| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597459845994600460146024603460446054606460746084609461046114612461346144615461646174618461946204621462246234624462546264627462846294630463146324633463446354636463746384639464046414642464346444645464646474648464946504651465246534654465546564657465846594660466146624663466446654666466746684669467046714672467346744675467646774678467946804681468246834684468546864687468846894690469146924693469446954696469746984699470047014702470347044705470647074708470947104711471247134714471547164717471847194720472147224723472447254726472747284729473047314732473347344735473647374738473947404741474247434744474547464747474847494750475147524753475447554756475747584759476047614762476347644765476647674768476947704771477247734774477547764777477847794780478147824783478447854786478747884789479047914792479347944795479647974798479948004801480248034804480548064807480848094810481148124813481448154816481748184819482048214822482348244825482648274828482948304831483248334834483548364837483848394840484148424843484448454846484748484849485048514852485348544855485648574858485948604861486248634864486548664867486848694870487148724873487448754876487748784879488048814882488348844885488648874888488948904891489248934894489548964897489848994900490149024903490449054906490749084909491049114912491349144915491649174918491949204921492249234924492549264927492849294930493149324933493449354936493749384939494049414942494349444945494649474948494949504951495249534954495549564957495849594960496149624963496449654966496749684969497049714972497349744975497649774978497949804981498249834984498549864987498849894990499149924993499449954996499749984999500050015002500350045005500650075008500950105011501250135014501550165017501850195020502150225023502450255026502750285029503050315032503350345035503650375038503950405041504250435044504550465047504850495050505150525053505450555056505750585059506050615062506350645065506650675068506950705071507250735074507550765077507850795080508150825083508450855086508750885089509050915092509350945095509650975098509951005101510251035104510551065107510851095110511151125113511451155116511751185119512051215122512351245125512651275128512951305131513251335134513551365137513851395140514151425143514451455146514751485149515051515152515351545155515651575158515951605161516251635164516551665167516851695170517151725173517451755176517751785179518051815182518351845185518651875188518951905191519251935194519551965197519851995200520152025203520452055206520752085209521052115212521352145215521652175218521952205221522252235224522552265227522852295230523152325233523452355236523752385239524052415242524352445245524652475248524952505251525252535254525552565257525852595260526152625263526452655266526752685269527052715272527352745275527652775278527952805281528252835284528552865287528852895290529152925293529452955296529752985299530053015302530353045305530653075308530953105311531253135314531553165317531853195320532153225323532453255326532753285329533053315332533353345335533653375338533953405341534253435344534553465347534853495350535153525353535453555356535753585359536053615362536353645365536653675368536953705371537253735374537553765377537853795380538153825383538453855386538753885389539053915392539353945395539653975398539954005401540254035404540554065407540854095410541154125413541454155416541754185419542054215422542354245425542654275428542954305431543254335434543554365437543854395440544154425443544454455446544754485449545054515452545354545455545654575458545954605461546254635464546554665467546854695470547154725473547454755476547754785479548054815482548354845485548654875488548954905491549254935494549554965497549854995500550155025503550455055506550755085509551055115512551355145515551655175518551955205521552255235524552555265527552855295530553155325533553455355536553755385539554055415542554355445545554655475548554955505551555255535554555555565557555855595560556155625563556455655566556755685569557055715572557355745575557655775578557955805581558255835584558555865587558855895590559155925593559455955596559755985599560056015602560356045605560656075608560956105611561256135614561556165617561856195620562156225623562456255626562756285629563056315632563356345635563656375638563956405641564256435644564556465647564856495650565156525653565456555656565756585659566056615662566356645665566656675668566956705671567256735674567556765677567856795680568156825683568456855686568756885689569056915692569356945695569656975698569957005701570257035704570557065707570857095710571157125713571457155716571757185719572057215722572357245725572657275728572957305731573257335734573557365737573857395740574157425743574457455746574757485749575057515752575357545755575657575758575957605761576257635764576557665767576857695770577157725773577457755776577757785779578057815782578357845785578657875788578957905791579257935794579557965797579857995800580158025803580458055806580758085809581058115812581358145815581658175818581958205821582258235824582558265827582858295830583158325833583458355836583758385839584058415842584358445845584658475848584958505851585258535854585558565857585858595860586158625863586458655866586758685869587058715872587358745875587658775878587958805881588258835884588558865887588858895890589158925893589458955896589758985899590059015902590359045905590659075908590959105911591259135914591559165917591859195920592159225923592459255926592759285929593059315932593359345935593659375938593959405941594259435944594559465947594859495950595159525953595459555956595759585959596059615962596359645965596659675968596959705971597259735974597559765977597859795980598159825983598459855986598759885989599059915992599359945995599659975998599960006001600260036004600560066007600860096010601160126013601460156016601760186019602060216022602360246025602660276028602960306031603260336034603560366037603860396040604160426043604460456046604760486049605060516052605360546055605660576058605960606061606260636064606560666067606860696070607160726073607460756076607760786079608060816082608360846085608660876088608960906091609260936094609560966097609860996100610161026103610461056106610761086109611061116112611361146115611661176118611961206121612261236124612561266127612861296130613161326133613461356136613761386139614061416142614361446145614661476148614961506151615261536154615561566157615861596160616161626163616461656166616761686169617061716172617361746175617661776178617961806181618261836184618561866187618861896190619161926193619461956196619761986199620062016202620362046205620662076208620962106211621262136214621562166217621862196220622162226223622462256226622762286229623062316232623362346235623662376238623962406241624262436244624562466247624862496250625162526253625462556256625762586259626062616262626362646265626662676268626962706271627262736274627562766277627862796280628162826283628462856286628762886289629062916292629362946295629662976298629963006301630263036304630563066307630863096310631163126313631463156316631763186319632063216322632363246325632663276328632963306331633263336334633563366337633863396340634163426343634463456346634763486349635063516352635363546355635663576358635963606361636263636364636563666367636863696370637163726373637463756376637763786379638063816382638363846385638663876388638963906391639263936394639563966397639863996400640164026403640464056406640764086409641064116412641364146415641664176418641964206421642264236424642564266427642864296430643164326433643464356436643764386439644064416442644364446445644664476448644964506451645264536454645564566457645864596460646164626463646464656466646764686469647064716472647364746475647664776478647964806481648264836484648564866487648864896490649164926493649464956496649764986499650065016502650365046505650665076508650965106511651265136514651565166517651865196520652165226523652465256526652765286529653065316532653365346535653665376538653965406541654265436544654565466547654865496550655165526553655465556556655765586559656065616562656365646565656665676568656965706571657265736574657565766577657865796580658165826583658465856586658765886589659065916592659365946595659665976598659966006601660266036604660566066607660866096610661166126613661466156616661766186619662066216622662366246625662666276628662966306631663266336634663566366637663866396640664166426643664466456646664766486649665066516652665366546655665666576658665966606661666266636664666566666667666866696670667166726673667466756676667766786679668066816682668366846685668666876688668966906691669266936694669566966697669866996700670167026703670467056706670767086709671067116712671367146715671667176718671967206721672267236724672567266727672867296730673167326733673467356736673767386739674067416742674367446745674667476748674967506751675267536754675567566757675867596760676167626763676467656766676767686769677067716772677367746775677667776778677967806781678267836784678567866787678867896790679167926793679467956796679767986799680068016802680368046805680668076808680968106811681268136814681568166817681868196820682168226823682468256826682768286829683068316832683368346835683668376838683968406841684268436844684568466847684868496850685168526853685468556856685768586859686068616862686368646865686668676868686968706871687268736874687568766877687868796880688168826883688468856886688768886889689068916892689368946895689668976898689969006901690269036904690569066907690869096910691169126913691469156916691769186919692069216922692369246925692669276928692969306931693269336934693569366937693869396940694169426943694469456946694769486949695069516952695369546955695669576958695969606961696269636964696569666967696869696970697169726973697469756976697769786979698069816982698369846985698669876988698969906991699269936994699569966997699869997000700170027003700470057006700770087009701070117012701370147015701670177018701970207021702270237024702570267027702870297030703170327033703470357036703770387039704070417042704370447045704670477048704970507051705270537054705570567057705870597060706170627063706470657066706770687069707070717072707370747075707670777078707970807081708270837084708570867087708870897090709170927093709470957096709770987099710071017102710371047105710671077108710971107111711271137114711571167117711871197120712171227123712471257126712771287129713071317132713371347135713671377138713971407141714271437144714571467147714871497150715171527153715471557156715771587159716071617162716371647165716671677168716971707171717271737174717571767177717871797180718171827183718471857186718771887189719071917192719371947195719671977198719972007201720272037204720572067207720872097210721172127213721472157216721772187219722072217222722372247225722672277228722972307231723272337234723572367237723872397240724172427243724472457246724772487249725072517252725372547255725672577258725972607261726272637264726572667267726872697270727172727273727472757276727772787279728072817282728372847285728672877288728972907291729272937294729572967297729872997300730173027303730473057306730773087309731073117312731373147315731673177318731973207321732273237324732573267327732873297330733173327333733473357336733773387339734073417342734373447345734673477348734973507351735273537354735573567357735873597360736173627363736473657366736773687369737073717372737373747375737673777378737973807381738273837384738573867387738873897390739173927393739473957396739773987399740074017402740374047405740674077408740974107411741274137414741574167417741874197420742174227423742474257426742774287429743074317432743374347435743674377438743974407441744274437444744574467447 | using Aitex.Common.Util;using Aitex.Core.Common;using Aitex.Core.RT.Device;using Aitex.Core.RT.Event;using Aitex.Core.RT.Routine;using Aitex.Core.RT.SCCore;using Aitex.Core.Util;using Aitex.Sorter.Common;using MECF.Framework.Common.Equipment;using MECF.Framework.Common.SubstrateTrackings;using System;using System.Collections.Generic;using System.IO;using System.Linq;using System.Text;using System.Threading.Tasks;using FurnaceRT.Equipments.CarrierRobots;using FurnaceRT.Equipments.PMs.RecipeExecutions;using FurnaceRT.Equipments.Schedulers;using FurnaceRT.Equipments.Systems;using FurnaceRT.Equipments.WaferRobots;using MECF.Framework.Common.Jobs;using Aitex.Core.RT.Log;using MECF.Framework.Common.DBCore;using Aitex.Core.RT.DataCenter;using System.Diagnostics;using FurnaceRT.Equipments.PMs;using MECF.Framework.Common.Schedulers;using System.Windows.Input;using System.Net.Sockets;using FurnaceRT.Equipments.FIMSs;using FurnaceRT.Devices;using System.Text.RegularExpressions;using MECF.Framework.Common.CommonData;using FurnaceRT.Equipments.Boats;using MECF.Framework.FA.Core.FAControl;using MECF.Framework.UI.Client.CenterViews.Operations.FA;using FurnaceRT.Equipments.LPs;using MECF.Framework.Common.CommonData.EnumData;using MECF.Framework.Common.DataCenter;using DocumentFormat.OpenXml.Drawing.Charts;using MECF.Framework.Common.Utilities;using System.Threading;using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.PMs;namespace FurnaceRT.Equipments.Jobs{    public class AutoTransfer : SchedulerModuleFactory    {        private List<ControlJobInfo> _lstControlJobs = new List<ControlJobInfo>();        private List<ProcessJobInfo> _lstProcessJobs = new List<ProcessJobInfo>();        public List<ProcessJobInfo> LstProcessJobs => _lstProcessJobs;        //private Object _locker = new Object();        private const string LogSource = "Scheduler";        System.Timers.Timer timer = null;        private bool _isCycleMode;        private int _cycleSetPoint = 0;        private int _cycledCount = 0;        private int _cycledWafer = 0;        private int _cycledTotalWafer = 0;        private int _cycleTime;        private int _cycleTimeSecond = 0;        private Stopwatch _coolTimer = new Stopwatch();        private SchedulerFACallback _faCallback;        private SchedulerDBCallback _dbCallback;        private ProcessJobInfo _secondPj;        private Dictionary<string, int> _sideDummyWafers = new Dictionary<string, int>();        private Dictionary<string, int> _extraDummyWafers = new Dictionary<string, int>();        private List<int> _sideDummySlots = new List<int>();        private List<int> _extraDummySlots = new List<int>();        private List<int> _productSlots = new List<int>();        private List<int> _monitor1Slots = new List<int>();        private List<int> _monitor2Slots = new List<int>();        private List<int> _tSlots = new List<int>();//expert mode        private List<int> _xdSlots = new List<int>();//expert mode        private List<SchedulerStocker> _lstCurrentPjStocker = new List<SchedulerStocker>();        private Dictionary<string, R_TRIG> _chargeRuleAlarmTrigDic = new Dictionary<string, R_TRIG>();        private Dictionary<string, R_TRIG> _waferStatusAlarmTrigDic = new Dictionary<string, R_TRIG>();        private Dictionary<string, R_TRIG> _dummyWaferCheckTrigDic = new Dictionary<string, R_TRIG>();        private Dictionary<string, R_TRIG> _logTrigDic = new Dictionary<string, R_TRIG>();        private bool _isCoolingSkip = false;        public bool HasJobRunning        {            get { return _lstControlJobs.Count > 0; }        }        public AutoTransfer()        {            _faCallback = new SchedulerFACallback();            _dbCallback = new SchedulerDBCallback();            InitModule();            _cycledTotalWafer = SC.ContainsItem("System.TotalCycledWafer")                ? SC.GetValue<int>("System.TotalCycledWafer")                : 0;            DATA.Subscribe("Scheduler.CycledCount", () => _cycledCount);            DATA.Subscribe("Scheduler.CycledWafer", () => _cycledWafer);            DATA.Subscribe("Scheduler.CycleSetPoint", () => _cycleSetPoint);            DATA.Subscribe("Scheduler.CycledTotalWafer", () => _cycledTotalWafer);            DATA.Subscribe("Scheduler.SecondPj", () => new List<ProcessJobInfo>() { _secondPj });            DATA.Subscribe("Scheduler.PjIdList", () => Array.ConvertAll(_lstProcessJobs.ToArray(), x => x.InnerId.ToString()).ToList());            DATA.Subscribe($"LP1.LocalJobName", () =>            {                var cj = _lstControlJobs.FirstOrDefault(x => x.Module == "LP1");                if (cj != null)                    return cj.Name;                return "";            });            DATA.Subscribe($"LP2.LocalJobStatus", () =>            {                var cj = _lstControlJobs.FirstOrDefault(x => x.Module == "LP2");                if (cj != null)                    return cj.State.ToString();                return "";            });            DATA.Subscribe($"Scheduler.CurrentControlJob", () =>            {                var cj = _lstControlJobs.Find(x => x.State == EnumControlJobState.Executing);                return cj;            });            DATA.Subscribe($"Scheduler.CurrentProcessJob", () =>            {                var pj = _lstProcessJobs.Find(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused);                return pj;            });            DATA.Subscribe($"Scheduler.ControlJobList", () =>            {                return _lstControlJobs;            });            DATA.Subscribe($"Scheduler.ProcessJobList", () =>            {                return _lstProcessJobs;            });            DATA.Subscribe($"Scheduler.CoolingTime", () =>            {                var pj = _lstProcessJobs.Find(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused);                return pj != null ? pj.CoolTimeSec : 0;            });            DATA.Subscribe($"Scheduler.CoolingRemainTime", () =>            {                var pj = _lstProcessJobs.Find(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused);                var coolTime = pj != null ? pj.CoolTimeSec : 0;                if (_coolTimer.IsRunning)                    return (coolTime - _coolTimer.ElapsedMilliseconds / 1000) > 0 ? coolTime - _coolTimer.ElapsedMilliseconds / 1000 : 0;                return coolTime;            });            DATA.Subscribe($"Scheduler.StockerUseSlots", () =>            {                Dictionary<string, string> stockerUseSlots = new Dictionary<string, string>();                if (!_lstProcessJobs.Any())                    return stockerUseSlots;                foreach (var pj in _lstProcessJobs)                {                    if (pj.Stockers != null)                    {                        foreach (var stocker in pj.Stockers)                        {                            if (!stockerUseSlots.ContainsKey(stocker.Item1))                                stockerUseSlots.Add(stocker.Item1, string.Join(",", stocker.Item3));                            else                            {                                var value = stockerUseSlots[stocker.Item1];                                stockerUseSlots[stocker.Item1] = $"{value},{string.Join(",", stocker.Item3)}";                            }                        }                    }                }                return stockerUseSlots;            });        }        public void Clear()        {            _carrierRobot.ResetTask();            _waferRobot.ResetTask();            foreach (var pm in _lstPms)            {                pm.ResetTask();            }            foreach (var lp in _lstLPs)            {                lp.ResetTask();            }            foreach (var fims in _lstFIMSs)            {                fims.ResetTask();            }            _lstControlJobs.Clear();            _lstProcessJobs.Clear();            _isCoolingSkip = false;        }        public void ResetTask(ModuleName module)        {            if (module == _carrierRobot.Module)                _carrierRobot.ResetTask();            if (module == _waferRobot.Module)                _waferRobot.ResetTask();            foreach (var lp in _lstLPs)            {                if (module == _carrierRobot.Module)                    lp.ResetTask();            }            foreach (var fims in _lstFIMSs)            {                if (module == _carrierRobot.Module)                    fims.ResetTask();            }        }        #region Job Management        public Result CreateJob(params object[] objs)        {            _isCycleMode = SC.GetValue<bool>("System.IsCycleMode");            _cycleSetPoint = _isCycleMode ? SC.GetValue<int>("System.CycleCount") : 0;            if (_isCycleMode)            {                _cycledWafer = 0;                _cycledCount = 0;            }            if (objs == null || objs.Length == 0)            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set("Parameter not valid");                return Result.FAIL;            }            var param = (Dictionary<string, object>)objs[0];            if (param == null)            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set("Parameter not valid");                return Result.FAIL;            }            var content = "";            foreach (var key in param.Keys)            {                content += $"{key}-{param[key]}#";            }            LOG.Write($"Control Job: {content}");            if (!Singleton<EquipmentManager>.Instance.GetJobRecipeInfor((string)param["JobRecipe"], out string processRecipe, out string layoutRecipe, out int coolTime, out string reason))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set(reason);                return Result.FAIL;            }            if (string.IsNullOrEmpty(layoutRecipe))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set("Parameter not valid, Layout Recipe is null");                return Result.FAIL;            }            var path = $"{PathManager.GetRecipeDir()}\\{SC.GetStringValue("System.Recipe.SupportedChamberType")}\\Layout\\{layoutRecipe}.rcp";            if (!File.Exists(path))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Layout \"{layoutRecipe}\" not exist.");                return Result.FAIL;            }            path = $"{PathManager.GetRecipeDir()}\\{SC.GetStringValue("System.Recipe.SupportedChamberType")}\\Process\\{processRecipe}.rcp";            if (!File.Exists(path))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Process recipe \"{processRecipe}\" not exist.");                return Result.FAIL;            }            if (!RecipeParser.Parse(processRecipe, ModuleName.PM1.ToString(), out var recipeHead, out var recipeSteps, out reason, "Process"))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Load process recipe {processRecipe} failed, {reason}");                return Result.FAIL;            }            List<Tuple<string, string, List<int>>> useStockers = new List<Tuple<string, string, List<int>>>();            if (RecipeParser.LayoutRecipeParse(layoutRecipe, ModuleName.PM1.ToString(), out RecipeLayoutEntityNormal layoutRecipeDataNormal, out RecipeLayoutEntityExpert layoutRecipeDataExpert, out reason))            {                if (param.ContainsKey("CarrierPara"))                {                    //Stocker1,P_00000036,25,Exist,1111111111111111111111111,LP1;Stocker2,P_00000037,25,Exist,1111111111111111111111111,LP2                    var paras = param["CarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        foreach (var item in paras)                        {                            //Stocker13,,21/22/23/24/25"                            var arry = item.Split(',');                            if (arry != null && arry.Length >= 2)                            {                                var pStocker = GetModule(arry[0]) as SchedulerStocker;                                if (pStocker != null && (SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("P") || SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("None")))                                {                                    var slots = new List<int>();                                    if (arry.Length >= 3)                                    {                                        var slotParas = arry[2].Split('/').ToList();                                        if (slotParas != null && slotParas.Count > 0)                                        {                                            foreach (var slotPara in slotParas)                                            {                                                var ret = int.TryParse(slotPara, out int slot);                                                if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                {                                                    if (_lstProcessJobs.Any())                                                    {                                                        foreach (var job in _lstProcessJobs)                                                        {                                                            if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                            {                                                                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                return Result.FAIL;                                                            }                                                        }                                                    }                                                    slots.Add(slot - 1);                                                }                                            }                                        }                                    }                                    useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                }                            }                        }                    }                }                if (param.ContainsKey("mCarrierPara"))                {                    //Stocker14,P_00000014;Stocker15,P_00000015                    var paras = param["mCarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        if (!string.IsNullOrEmpty(paras[0]))                        {                            foreach (var item in paras)                            {                                var arry = item.Split(',');                                if (arry != null && arry.Length >= 2)                                {                                    var mStocker = GetModule(arry[0]) as SchedulerStocker;                                    if (mStocker != null && SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("M"))                                    {                                        var slots = new List<int>();                                        if (arry.Length >= 3)                                        {                                            var slotParas = arry[2].Split('/').ToList();                                            if (slotParas != null && slotParas.Count > 0)                                            {                                                foreach (var slotPara in slotParas)                                                {                                                    var ret = int.TryParse(slotPara, out int slot);                                                    if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                    {                                                        if (_lstProcessJobs.Any())                                                        {                                                            foreach (var job in _lstProcessJobs)                                                            {                                                                if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                                {                                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                    return Result.FAIL;                                                                }                                                            }                                                        }                                                        slots.Add(slot - 1);                                                    }                                                }                                            }                                        }                                        useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                    }                                }                            }                        }                    }                }                if (param.ContainsKey("m1CarrierPara"))                {                    //Stocker14,P_00000014;Stocker15,P_00000015                    var paras = param["m1CarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        if (!string.IsNullOrEmpty(paras[0]))                        {                            foreach (var item in paras)                            {                                var arry = item.Split(',');                                if (arry != null && arry.Length >= 2)                                {                                    var mStocker = GetModule(arry[0]) as SchedulerStocker;                                    if (mStocker != null && (SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("M1") || SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("None")))                                    {                                        var slots = new List<int>();                                        if (arry.Length >= 3)                                        {                                            var slotParas = arry[2].Split('/').ToList();                                            if (slotParas != null && slotParas.Count > 0)                                            {                                                foreach (var slotPara in slotParas)                                                {                                                    var ret = int.TryParse(slotPara, out int slot);                                                    if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                    {                                                        if (_lstProcessJobs.Any())                                                        {                                                            foreach (var job in _lstProcessJobs)                                                            {                                                                if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                                {                                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                    return Result.FAIL;                                                                }                                                            }                                                        }                                                        slots.Add(slot - 1);                                                    }                                                }                                            }                                        }                                        useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                    }                                }                            }                        }                    }                }                if (param.ContainsKey("m2CarrierPara"))                {                    //Stocker14,P_00000014;Stocker15,P_00000015                    var paras = param["m2CarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        if (!string.IsNullOrEmpty(paras[0]))                        {                            foreach (var item in paras)                            {                                var arry = item.Split(',');                                if (arry != null && arry.Length >= 2)                                {                                    var mStocker = GetModule(arry[0]) as SchedulerStocker;                                    if (mStocker != null && (SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("M2") || SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("None")))                                    {                                        var slots = new List<int>();                                        if (arry.Length >= 3)                                        {                                            var slotParas = arry[2].Split('/').ToList();                                            if (slotParas != null && slotParas.Count > 0)                                            {                                                foreach (var slotPara in slotParas)                                                {                                                    var ret = int.TryParse(slotPara, out int slot);                                                    if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                    {                                                        if (_lstProcessJobs.Any())                                                        {                                                            foreach (var job in _lstProcessJobs)                                                            {                                                                if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                                {                                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                    return Result.FAIL;                                                                }                                                            }                                                        }                                                        slots.Add(slot - 1);                                                    }                                                }                                            }                                        }                                        useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                    }                                }                            }                        }                    }                }                useStockers.Sort(compare);                foreach (var s in _lstSideDummyStocker)                {                    if (CarrierManager.Instance.CheckHasCarrier(s.Module, 0) && CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierType == CarrierType.SD)                        useStockers.Add(Tuple.Create(s.Module.ToString(), CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierId, new List<int>()));                    else                    {                        bool onRobot = false;                        if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))                        {                            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);                            if (carrier.CarrierType == CarrierType.SD &&                                carrier.InternalModuleName == s.Module)                            {                                useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                                onRobot = true;                            }                        }                        if (!onRobot)                        {                            bool inFIMS = false;                            foreach (var fims in _lstFIMSs)                            {                                if (!fims.IsOnline)                                    continue;                                var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                                if (!carrier.IsEmpty && carrier.CarrierType == CarrierType.SD &&                                    carrier.InternalModuleName == s.Module)                                {                                    inFIMS = true;                                    useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                                }                            }                            if (!inFIMS && CarrierManager.Instance.GetLocationByInternalCarrierModuleName(s.Module, CarrierType.SD) != null)                                useStockers.Add(Tuple.Create(s.Module.ToString(), "", new List<int>()));                        }                    }                }                foreach (var s in _lstExtraDummyStocker)                {                    if (CarrierManager.Instance.CheckHasCarrier(s.Module, 0) && CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierType == CarrierType.ED)                        useStockers.Add(Tuple.Create(s.Module.ToString(), CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierId, new List<int>()));                    else                    {                        bool onRobot = false;                        if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))                        {                            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);                            if (carrier.CarrierType == CarrierType.ED &&                                carrier.InternalModuleName == s.Module)                            {                                useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                                onRobot = true;                            }                        }                        if (!onRobot)                        {                            bool inFIMS = false;                            foreach (var fims in _lstFIMSs)                            {                                if (!fims.IsOnline)                                    continue;                                var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                                if (!carrier.IsEmpty && carrier.CarrierType == CarrierType.ED &&                                    carrier.InternalModuleName == s.Module)                                {                                    inFIMS = true;                                    useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                                }                            }                            if (!inFIMS && CarrierManager.Instance.GetLocationByInternalCarrierModuleName(s.Module, CarrierType.SD) != null)                                useStockers.Add(Tuple.Create(s.Module.ToString(), "", new List<int>()));                        }                    }                }                var wafers = WaferManager.Instance.GetWafers(_pm1.Module);                if (wafers.Any(x => !x.IsEmpty) && !_lstProcessJobs.Any(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused) &&                    SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have wafer in boat");                    return Result.FAIL;                }                if (wafers.Any(x => !x.IsEmpty && x.WaferType == WaferType.None))                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have unknown wafer in boat");                    return Result.FAIL;                }                var robotWafers = WaferManager.Instance.GetWafers(ModuleName.WaferRobot);                if (robotWafers.Any(x => !x.IsEmpty) && !_lstProcessJobs.Any(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused))                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have wafer in wafer robot");                    return Result.FAIL;                }                if (robotWafers.Any(x => !x.IsEmpty && x.WaferType == WaferType.None))                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have unknown wafer in wafer robot");                    return Result.FAIL;                }                if (!CheckWaferStatus(useStockers, true, true, out reason))                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set(reason);                    return Result.FAIL;                }            }            var pjJobId = param["PJID"].ToString();            var cjJobId = param["CJID"].ToString();            ControlJobInfo cj = null;            if (_lstControlJobs.Exists(x => x.Name == cjJobId))            {                for (int i = 1; i < 1000; i++)                {                    if (!_lstControlJobs.Exists(x => x.Name == $"{cjJobId}_{i}"))                    {                        cjJobId = $"{cjJobId}_{i}";                        break;                    }                }            }            cj = new ControlJobInfo();            cj.Name = cjJobId;            //cj.Module = module;            cj.LotName = cjJobId;            cj.LotInnerId = Guid.NewGuid();            cj.LotWafers = new List<WaferInfo>();            cj.CreateTime = DateTime.Now;            cj.SetState(EnumControlJobState.WaitingForStart);            ProcessJobInfo pj = new ProcessJobInfo();            pj.Name = pjJobId;            pj.JobRecipe = param["JobRecipe"].ToString();            pj.ProcessRecipe = processRecipe;            pj.LayoutRecipe = layoutRecipe;            pj.ControlJobName = cj.Name;            pj.SetState(EnumProcessJobState.Queued);            pj.CreateTime = DateTime.Now;            pj.Stockers = useStockers;            pj.SlotWafers = new List<Tuple<ModuleName, int>>();            pj.ProcessingState = EnumProcessingState.WaitingMaterial;            pj.CoolTimeSec = coolTime;            pj.IsEmptyLayout = !layoutRecipeDataExpert.Items.Any(x => x != "----");            pj.IsN2PurgeMode = recipeHead.IsN2PurgeMode;            pj.N2PurgeModeStr = recipeHead.N2PurgeModeStr;            if (useStockers.Any() && !CheckChargeRule(pj, out bool waferShort, false, true, out reason))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {layoutRecipe} check charge rule failed, {reason}");                return Result.FAIL;            }            foreach (var stocker in useStockers)            {                if (CarrierManager.Instance.CheckHasCarrier($"{stocker.Item1}", 0) &&                    CarrierManager.Instance.GetCarrier($"{stocker.Item1}", 0).InternalModuleName.ToString() != stocker.Item1)                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Carrier of {stocker.Item1} is not match stocker module name {CarrierManager.Instance.GetCarrier($"{stocker.Item1}", 0).InternalModuleName}");                    return Result.FAIL;                }            }            _lstProcessJobs.Add(pj);            cj.ProcessJobNameList.Add(pj.Name);            if (!_lstControlJobs.Contains(cj))                _lstControlJobs.Add(cj);            _faCallback.JobCreated(cj, pj);            _secondPj = null;            if ((Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule).IsExcuteIdleRecipe)                (Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule).EndIdleRecipe();            return Result.RUN;        }        static int compare(Tuple<string, string> t1, Tuple<string, string> t2)        {            int.TryParse(t1.Item1.Replace("Stocker", ""), out int s1);            int.TryParse(t2.Item1.Replace("Stocker", ""), out int s2);            if (s1 > s2)            {                return 1;            }            else            {                return -1;            }        }        static int compare(Tuple<string, string, List<int>> t1, Tuple<string, string, List<int>> t2)        {            int.TryParse(t1.Item1.Replace("Stocker", ""), out int s1);            int.TryParse(t2.Item1.Replace("Stocker", ""), out int s2);            if (s1 > s2)            {                return 1;            }            else            {                return -1;            }        }        internal ProcessJobInfo GetFirstProcessJob(ControlJobInfo cj, ProcessJobInfo checkPj = null)        {            DateTime dtMin = DateTime.MaxValue;            ProcessJobInfo currentPj = null;            foreach (var pj in _lstProcessJobs)            {                if (checkPj != null && checkPj == pj)                    continue;                if (pj.ControlJobName == cj.Name)                {                    if (pj.CreateTime.CompareTo(dtMin) < 0)                    {                        dtMin = pj.CreateTime;                        currentPj = pj;                    }                }            }            return currentPj;        }        internal void StopJob(string jobName)        {            ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);            if (cj == null)            {                EV.PostWarningLog(LogSource, $"stop job rejected, not found job with id {jobName}");                return;            }            foreach (var pj in _lstProcessJobs)            {                if (pj.ControlJobName == cj.Name)                {                    if (pj.State == EnumProcessJobState.Processing)                        _faCallback.JobStopped(cj, pj);                    pj.SetState(EnumProcessJobState.Stopping);                    pj.ProcessingState = EnumProcessingState.WaitingWithdrawal;                }            }            if (_isCycleMode)            {                if (_cycleSetPoint > _cycledCount)                {                    _cycleSetPoint = _cycledCount;                    EV.PostWarningLog("System", $"system in cycle mode, cycle count reset to {_cycledCount}");                }            }            _dbCallback.LotFinished(cj);        }        internal void AbortJob(string jobName)        {            ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);            if (cj == null)            {                EV.PostWarningLog(LogSource, $"abort job rejected, not found job with id {jobName}");                return;            }            List<ProcessJobInfo> pjAbortList = new List<ProcessJobInfo>();            foreach (var pj in _lstProcessJobs)            {                if (pj.ControlJobName == cj.Name)                {                    if (pj.State == EnumProcessJobState.Processing)                        _faCallback.JobAborted(cj, pj);                    pj.SetState(EnumProcessJobState.Aborting);                    if (pj.ProcessingState == EnumProcessingState.Charging || pj.ProcessingState == EnumProcessingState.Discharging)                    {                        var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                        pmModule?.SetN2PurgeLAO2CheckFirstEnable(false);                        pmModule?.SetN2PurgeProcess(false);                    }                    pj.ProcessingState = EnumProcessingState.WaitingWithdrawal;                    pjAbortList.Add(pj);                    int unprocessed = 0;                    int aborted = 0;                    WaferInfo[] wafers = WaferManager.Instance.GetWaferByProcessJob(pj.Name);                    foreach (var waferInfo in wafers)                    {                        waferInfo.ProcessJob = null;                        waferInfo.NextSequenceStep = 0;                        if (waferInfo.ProcessState != EnumWaferProcessStatus.Completed)                            unprocessed++;                    }                    JobDataRecorder.EndPJ(pj.InnerId.ToString(), aborted, unprocessed);                }            }            Singleton<FAJobController>.Instance.FAProcessJobs[jobName].RequestAbort();            foreach (var pj in pjAbortList)            {                _lstProcessJobs.Remove(pj);            }            _lstControlJobs.Remove(cj);            _dbCallback.LotFinished(cj);        }        public void ResumeJob(string jobName)        {            ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);            if (cj == null)            {                EV.PostWarningLog(LogSource, $"resume job rejected, not found job with id {jobName}");                return;            }            if (cj.State == EnumControlJobState.Paused)            {                cj.SetState(EnumControlJobState.Executing);            }            foreach (var pj in _lstProcessJobs)            {                if (pj.ControlJobName == cj.Name)                {                    if (pj.State == EnumProcessJobState.Processing)                        _faCallback.JobResumed(cj, pj);                }            }        }        internal bool StartJob(string jobName)        {            if (!CheckPMStatus())            {                EV.PostWarningLog("Scheduler", "can not change to auto mode, at least one process chamber be online");                return false;            }            ControlJobInfo cj = _lstControlJobs.Find(x => x.Name == jobName);            if (cj == null)            {                EV.PostWarningLog(LogSource, $"start job rejected, not found job with id {jobName}");                return false;            }            if (_isCycleMode)            {                if (CheckAllJobDone())                {                    _cycledWafer = 0;                    _cycledCount = 0;                }            }            if (cj.State == EnumControlJobState.WaitingForStart)            {                cj.SetState(EnumControlJobState.Executing);                //(GetModule(cj.Module) as SchedulerLoadPort).NoteJobStart();                cj.LotInnerId = Guid.NewGuid();                cj.BeginTime = DateTime.Now;                _dbCallback.LotCreated(cj);                _faCallback.JobStarted(cj, GetFirstProcessJob(cj));            }            return true;        }        internal void PauseJob(string jobName)        {            foreach (var cj in _lstControlJobs)            {                if (cj.Name != jobName && !string.IsNullOrEmpty(jobName))                    continue;                if (cj.State == EnumControlJobState.Executing)                {                    cj.SetState(EnumControlJobState.Paused);                    _faCallback.JobPaused(cj, GetFirstProcessJob(cj));                }            }        }        #endregion Job Management        private bool CheckPMStatus()        {            bool hasPmOnline = false;            foreach (var schedulerPm in _lstPms)            {                if (schedulerPm.IsAvailable)                {                    hasPmOnline = true;                    continue;                }            }            return hasPmOnline;        }        public Result Start(params object[] objs)        {            Clear();            _cycledWafer = 0;            _cycledCount = 0;            timer.Start();            if (!CheckPMStatus())            {                EV.PostWarningLog("Scheduler", "can not change to auto mode, at least one process chamber be online");                return Result.FAIL;            }            return Result.RUN;        }        public Result Monitor()        {            ControlJobInfo cjActive = _lstControlJobs.Find(x => x.State == EnumControlJobState.Executing);            if (cjActive != null)            {                MonitorModuleTasks();            }            MonitorJobTasks();            return Result.RUN;        }        #region Job task        public Result MonitorJobTasks()        {            UpdateProcessJobStatus();            UpdateControlJobStatus();            StartNewJob();            return Result.RUN;        }        private void UpdateProcessJobStatus()        {            foreach (var pj in _lstProcessJobs)            {                if (pj.ProcessingState == EnumProcessingState.Discharging && pj.IsEmptyLayout)                {                    pj.SetState(EnumProcessJobState.ProcessingComplete);                    JobDataRecorder.EndPJ(pj.InnerId.ToString(), 0, 0);                    var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                    pmModule?.SetN2PurgeLAO2CheckFirstEnable(false);                    pmModule?.SetN2PurgeProcess(false);                }                else                {                    if (pj.State == EnumProcessJobState.Processing)                    {                        if (CheckAllWaferReturned(pj, true))                        {                            pj.SetState(EnumProcessJobState.ProcessingComplete);                            JobDataRecorder.EndPJ(pj.InnerId.ToString(), 0, 0);                            var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                            pmModule?.SetN2PurgeLAO2CheckFirstEnable(false);                            pmModule?.SetN2PurgeProcess(false);                        }                    }                    else if (pj.State == EnumProcessJobState.Stopping)                    {                        if (CheckAllWaferReturned(pj, false))                        {                            pj.SetState(EnumProcessJobState.ProcessingComplete);                            JobDataRecorder.EndPJ(pj.InnerId.ToString(), 0, 0);                            var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                            pmModule?.SetN2PurgeLAO2CheckFirstEnable(false);                            pmModule?.SetN2PurgeProcess(false);                        }                    }                }            }        }        private void UpdateControlJobStatus()        {            if (_lstControlJobs.Count == 0)                return;            bool allControlJobComplete = true;            List<ControlJobInfo> cjRemoveList = new List<ControlJobInfo>();            foreach (var cj in _lstControlJobs)            {                if (cj.State == EnumControlJobState.Executing)                {                    bool allPjCompleted = true;                    foreach (var pjName in cj.ProcessJobNameList)                    {                        var pj = _lstProcessJobs.Find(x => x.Name == pjName);                        if (pj == null)                        {                            LOG.Error($"Not find pj named {pjName} in {cj.Name}");                            continue;                        }                        if (pj.State != EnumProcessJobState.Complete && pj.State != EnumProcessJobState.ProcessingComplete)                        {                            allPjCompleted = false;                            break;                        }                    }                    if (allPjCompleted)                    {                        cj.SetState(EnumControlJobState.Completed);                        cj.EndTime = DateTime.Now;                        string strInfo = $"{cj.Module} has finished: \r\n\r\nProcessed wafer number: {cj.LotWafers.Count} \r\nBegin time: {cj.BeginTime} \r\nEnd time: {cj.EndTime} \r\nDuration: {(cj.EndTime - cj.BeginTime).TotalSeconds:F0} sec";                        EV.PostInfoLog("Scheduler", strInfo);                        if (SC.GetValue<bool>("System.Job.EnablePopDialogWhenJobDone"))                        {                            EV.PostPopDialogMessage(EventLevel.Information, "System Information", strInfo);                        }                        _faCallback.JobFinished(cj, GetFirstProcessJob(cj));                        _dbCallback.LotFinished(cj);                        //if (!(_isCycleMode && _cycledCount < _cycleSetPoint))                        //    (GetModule(cj.Module) as SchedulerLoadPort).NoteJobComplete();                    }                }                if (cj.State == EnumControlJobState.Completed && (!SC.GetValue<bool>("System.IsCycleMode") ||                                                                 (SC.GetValue<bool>("System.IsCycleMode") && _cycledCount >= _cycleSetPoint)))                {                    cjRemoveList.Add(cj);                }                allControlJobComplete = allControlJobComplete && cj.State == EnumControlJobState.Completed;            }            if (_isCycleMode && _cycledCount < _cycleSetPoint)            {                int countPerCycle = 0;                int countProcessed = 0;                foreach (var pj in _lstProcessJobs)                {                    if (pj.State == EnumProcessJobState.Queued || pj.SlotWafers == null)                        continue;                    foreach (var pjSlotWafer in pj.SlotWafers)                    {                        countPerCycle++;                        WaferInfo wafer = WaferManager.Instance.GetWafer(pjSlotWafer.Item1, pjSlotWafer.Item2);                        if (!wafer.IsEmpty && !CheckWaferNeedProcess(pjSlotWafer.Item1, pjSlotWafer.Item2))                        {                            countProcessed++;                        }                    }                }                _cycledWafer = _cycledCount * countPerCycle + countProcessed;                if (allControlJobComplete)                {                    _cycledCount++;                    if (_cycledCount < _cycleSetPoint)                    {                        foreach (var cj in _lstControlJobs)                        {                            cj.SetState(EnumControlJobState.Executing);                            cj.LotInnerId = Guid.NewGuid();                            //_dbCallback.LotCreated(cj);                        }                        foreach (var pj in _lstProcessJobs)                        {                            pj.SetState(EnumProcessJobState.Queued);                            pj.ProcessingState = EnumProcessingState.WaitingMaterial;                            pj.InnerId = Guid.NewGuid();                            if (pj.SlotWafers == null)                                continue;                            foreach (var pjSlotWafer in pj.SlotWafers)                            {                                WaferInfo wafer = WaferManager.Instance.GetWafer(pjSlotWafer.Item1, pjSlotWafer.Item2);                                wafer.ProcessJob = null;                                wafer.NextSequenceStep = 0;                                wafer.ProcessState = EnumWaferProcessStatus.Idle;                            }                            pj.SlotWafers.Clear();                        }                    }                    else                    {                        foreach (var cj in _lstControlJobs)                        {                            if (cj.State == EnumControlJobState.Completed)                            {                                cjRemoveList.Add(cj);                            }                        }                    }                }            }            if (_waferRobot.IsAvailable && cjRemoveList.Any())            {                (Singleton<EquipmentManager>.Instance.Modules[ModuleName.WaferRobot] as WaferRobotModule).Home(out _);            }            foreach (var cj in cjRemoveList)            {                List<ProcessJobInfo> pjRemoveList = new List<ProcessJobInfo>();                foreach (var pj in _lstProcessJobs)                {                    if (pj.ControlJobName == cj.Name)                        pjRemoveList.Add(pj);                }                foreach (var pj in pjRemoveList)                {                    _lstProcessJobs.Remove(pj);                }                _lstControlJobs.Remove(cj);            }        }        public bool CheckAllJobDone()        {            foreach (var cj in _lstControlJobs)            {                if (cj.State == EnumControlJobState.Executing ||                    cj.State == EnumControlJobState.Paused ||                    cj.State == EnumControlJobState.Queued ||                    cj.State == EnumControlJobState.WaitingForStart)                    return false;            }            if (_lstControlJobs.Count > 0 || _lstProcessJobs.Count > 0)            {                return false;            }            return true;        }        public List<Tuple<string, string, List<int>>> GetProcessingStockers()        {            if (!_lstControlJobs.Any() || !_lstProcessJobs.Any())                return null;            foreach (var cj in _lstControlJobs)            {                if (cj.State == EnumControlJobState.Completed)                    continue;                var pj = GetFirstProcessJob(cj);                if (pj != null)                {                    if (pj.ProcessingState != EnumProcessingState.Discharging)                        return pj.Stockers;                    else                    {                        List<Tuple<string, string, List<int>>> temp = new List<Tuple<string, string, List<int>>>();                        foreach (var item in pj.Stockers)                        {                            if (CarrierManager.Instance.CheckHasCarrier(item.Item1, 0))                            {                                var wafers = WaferManager.Instance.GetWafers(ModuleName.PM1).ToList();                                if (wafers != null && !wafers.Any(x => !x.IsEmpty && x.OriginStation == (int)ModuleHelper.Converter(item.Item1)))                                    continue;                            }                            temp.Add(Tuple.Create(item.Item1, item.Item2, item.Item3));                        }                        return temp;                    }                }            }            return null;        }        public List<Tuple<string, string, List<int>>> GetSecondlyProcessingStockers()        {            var pj = GetSecondlyProcessJob();            return pj != null ? pj.Stockers : null;        }        private ProcessJobInfo GetSecondlyProcessJob()        {            if (!_lstControlJobs.Any() || !_lstProcessJobs.Any())                return null;            if (_lstControlJobs.Count > 1)            {                var cj = _lstControlJobs[1];                if (cj.State == EnumControlJobState.Completed)                    return null;                var pj = GetFirstProcessJob(cj);                return pj;            }            return null;        }        private ProcessJobInfo GetSecondlyProcessJobV2()        {            if (!_lstControlJobs.Any() || !_lstProcessJobs.Any())                return null;            if (_lstControlJobs.Count > 1 && _secondPj == null)            {                var cj = _lstControlJobs[1];                if (cj.State == EnumControlJobState.Completed)                    return null;                var pj = GetFirstProcessJob(cj);                return pj;            }            return _secondPj;        }        public List<Tuple<string, string, List<int>>> GetAllPJStockers()        {            List<Tuple<string, string, List<int>>> temp = new List<Tuple<string, string, List<int>>>();            if (!_lstControlJobs.Any() || !_lstProcessJobs.Any())                return temp;            foreach (var cj in _lstControlJobs)            {                if (cj.State == EnumControlJobState.Completed)                    continue;                foreach (var pj in _lstProcessJobs)                {                    if (pj != null && pj.ControlJobName == cj.Name)                    {                        temp.AddRange(pj.Stockers);                    }                }            }            return temp;        }        private void StartNewJob()        {            if (!_lstControlJobs.Any() || !_lstProcessJobs.Any())                return;            _secondPj = GetSecondlyProcessJobV2();            ControlJobInfo cjActived = null;            DateTime dtMin = DateTime.MaxValue;            foreach (var cj in _lstControlJobs)            {                if (cj.State == EnumControlJobState.Completed)                    continue;                if (cj.CreateTime.CompareTo(dtMin) < 0)                {                    dtMin = cj.CreateTime;                    cjActived = cj;                }            }            if (cjActived == null)                return;            if (cjActived.State != EnumControlJobState.Executing &&                cjActived.State != EnumControlJobState.Paused)            {                cjActived.BeginTime = DateTime.Now;                cjActived.SetState(EnumControlJobState.Executing);            }            ProcessJobInfo pj = null;            foreach (var cj in _lstControlJobs)            {                if (cj.State != EnumControlJobState.Executing)                    continue;                cjActived = cj;                pj = GetFirstProcessJob(cjActived);                if (pj != null && pj.State == EnumProcessJobState.Queued)                {                    if (!pj.IsEmptyLayout)                    {                        if (CheckProcessJobReady(pj))                        {                            //先不管charge rule                            if (!CheckChargeRule(pj, out bool waferShort, true, true, out string reason))                            {                                if (!_chargeRuleAlarmTrigDic.ContainsKey(pj.InnerId.ToString()))                                    _chargeRuleAlarmTrigDic.Add(pj.InnerId.ToString(), new R_TRIG());                                _chargeRuleAlarmTrigDic[pj.InnerId.ToString()].CLK = true;                                if (_chargeRuleAlarmTrigDic[pj.InnerId.ToString()].Q)                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {pj.LayoutRecipe} check charge rule failed, {reason}");                                return;                            }                            if (!CheckWaferStatus(pj.Stockers, false, false, out reason))                            {                                if (!_waferStatusAlarmTrigDic.ContainsKey(pj.InnerId.ToString()))                                    _waferStatusAlarmTrigDic.Add(pj.InnerId.ToString(), new R_TRIG());                                _waferStatusAlarmTrigDic[pj.InnerId.ToString()].CLK = true;                                if (_waferStatusAlarmTrigDic[pj.InnerId.ToString()].Q)                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set(reason);                                return;                            }                            if (!CheckDummyWaferUse(pj, cj))                                return;                            if (!SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                            {                                var wafers = WaferManager.Instance.GetWafers(ModuleName.PM1).ToList();                                if (wafers != null && wafers.Any(x => !x.IsEmpty && (x.WaferType == WaferType.SD || x.WaferType == WaferType.ED)))                                {                                    for (int i = 0; i < wafers.Count; i++)                                    {                                        var wafer = wafers[i];                                        if (wafer == null || wafer.IsEmpty)                                            continue;                                        if (wafer.WaferType != WaferType.SD && wafer.WaferType != WaferType.ED)                                        {                                            if (!_dummyWaferCheckTrigDic.ContainsKey(pj.InnerId.ToString()))                                                _dummyWaferCheckTrigDic.Add(pj.InnerId.ToString(), new R_TRIG());                                            _dummyWaferCheckTrigDic[pj.InnerId.ToString()].CLK = true;                                            if (_dummyWaferCheckTrigDic[pj.InnerId.ToString()].Q)                                                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {pj.LayoutRecipe} check charge rule failed, invalid wafer in boat");                                            return;                                        }                                        if (!pj.SideDummySlots.Any(x => x == i) && !pj.ExtraDummySlots.Any(x => x == i))                                        {                                            if (!_dummyWaferCheckTrigDic.ContainsKey(pj.InnerId.ToString()))                                                _dummyWaferCheckTrigDic.Add(pj.InnerId.ToString(), new R_TRIG());                                            _dummyWaferCheckTrigDic[pj.InnerId.ToString()].CLK = true;                                            if (_dummyWaferCheckTrigDic[pj.InnerId.ToString()].Q)                                                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {pj.LayoutRecipe} check charge rule failed, invalid dummy wafer in boat, dummy wafer layout mismatch");                                            return;                                        }                                    }                                    var existSDSlots = new List<int>();                                    foreach (var slot in pj.SideDummySlots)                                    {                                        var wafer = WaferManager.Instance.GetWafer(ModuleName.PM1, slot);                                        if (!wafer.IsEmpty)                                        {                                            if (wafer.WaferType != WaferType.SD)                                            {                                                if (!_dummyWaferCheckTrigDic.ContainsKey(pj.InnerId.ToString()))                                                    _dummyWaferCheckTrigDic.Add(pj.InnerId.ToString(), new R_TRIG());                                                _dummyWaferCheckTrigDic[pj.InnerId.ToString()].CLK = true;                                                if (_dummyWaferCheckTrigDic[pj.InnerId.ToString()].Q)                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {pj.LayoutRecipe} check charge rule failed, invalid dummy wafer in boat, SD wafer layout mismatch");                                                return;                                            }                                            else                                            {                                                existSDSlots.Add(slot);                                            }                                        }                                    }                                    var existEDSlots = new List<int>();                                    foreach (var slot in pj.ExtraDummySlots)                                    {                                        var wafer = WaferManager.Instance.GetWafer(ModuleName.PM1, slot);                                        if (!wafer.IsEmpty)                                        {                                            if (wafer.WaferType != WaferType.ED)                                            {                                                if (!_dummyWaferCheckTrigDic.ContainsKey(pj.InnerId.ToString()))                                                    _dummyWaferCheckTrigDic.Add(pj.InnerId.ToString(), new R_TRIG());                                                _dummyWaferCheckTrigDic[pj.InnerId.ToString()].CLK = true;                                                if (_dummyWaferCheckTrigDic[pj.InnerId.ToString()].Q)                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {pj.LayoutRecipe} check charge rule failed, invalid dummy wafer in boat, ED wafer layout mismatch");                                                return;                                            }                                            else                                            {                                                existEDSlots.Add(slot);                                            }                                        }                                    }                                    for (int i = 0; i < wafers.Count; i++)                                    {                                        var wafer = wafers[i];                                        if (wafer == null || wafer.IsEmpty)                                            continue;                                        wafer.ProcessState = EnumWaferProcessStatus.Idle;                                    }                                    pj.IsNeedChargeDummyWafer = pj.ExtraDummySlots.Count != existEDSlots.Count || pj.SideDummySlots.Count != existSDSlots.Count;                                }                                else                                {                                    pj.IsNeedChargeDummyWafer = true;                                }                            }                            else                            {                                pj.IsNeedChargeDummyWafer = true;                            }                            ActiveProcessJob(cjActived, pj);                            _sideDummyWafers = GetDummyWafers(_lstSideDummyStocker, pj.SideDummySlots.Count(), WaferType.SD);                            _extraDummyWafers = GetDummyWafers(_lstExtraDummyStocker, pj.ExtraDummySlots.Count(), WaferType.ED);                            _sideDummySlots = pj.SideDummySlots;                            _extraDummySlots = pj.ExtraDummySlots;                            _productSlots = pj.ProductSlots;                            _monitor1Slots = pj.Monitor1Slots;                            _monitor2Slots = pj.Monitor2Slots;                            _lstCurrentPjStocker.Clear();                            foreach (var s in pj.Stockers)                            {                                var module = GetModule(s.Item1) as SchedulerStocker;                                if (module != null)                                    _lstCurrentPjStocker.Add(module);                            }                        }                    }                    else                    {                        if (!SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        {                            var wafers = WaferManager.Instance.GetWafers(ModuleName.PM1).ToList();                            for (int i = 0; i < wafers.Count; i++)                            {                                var wafer = wafers[i];                                if (wafer == null || wafer.IsEmpty)                                    continue;                                if (wafer.WaferType != WaferType.ED && wafer.WaferType != WaferType.SD)                                    continue;                                wafer.ProcessState = EnumWaferProcessStatus.Idle;                            }                        }                        ActiveProcessJob(cjActived, pj);                    }                    break;                }            }            if (_lstControlJobs.Count > 1 && pj != null)            {                var second = GetSecondlyProcessJob();                if (second != null && second.IsEmptyLayout && !second.IsStandbyStarted &&                    (pj.ProcessingState == EnumProcessingState.Cooling || pj.ProcessingState == EnumProcessingState.Discharging))                {                    foreach (var pm in _lstPms)                    {                        if (pm.IsAvailable)                        {                            _dbCallback.PjCreated(second);                            _dbCallback.PjStart();                            pm.Standby(second.ProcessRecipe);                            second.IsStandbyStarted = true;                        }                    }                }            }        }        private bool ActiveProcessJob(ControlJobInfo cj, ProcessJobInfo pj)        {            //_dbCallback.JobCreated(cj, pj);            pj.SetState(EnumProcessJobState.Processing);            pj.ProcessingState = EnumProcessingState.Charging;            pj.BeginTime = DateTime.Now;            if (pj.IsN2PurgeMode)            {                var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                //pmModule?.SetN2PurgeProcess(true);                //if (SC.ContainsItem("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable") && SC.GetValue<bool>("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable"))                //    pmModule?.SetN2PurgeLAO2CheckFirstEnable(true);                if (SC.ContainsItem("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable") && SC.GetValue<bool>("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable"))                {                    pmModule?.SetN2PurgeProcess(true);                    pmModule?.SetN2PurgeLAO2CheckFirstEnable(true);                }                if (SC.ContainsItem("PM1.RecipeRelevancyN2Purge") && SC.GetValue<bool>("PM1.RecipeRelevancyN2Purge") && SC.ContainsItem("PM1.SelectN2PurgeMode"))                {                    var setValue = pmModule.GetN2PurgeModeEnumByStr(pj.N2PurgeModeStr);                    SC.SetItemValueFromString("PM1.SelectN2PurgeMode", setValue);                    pmModule.RestAllN2PrugeRD();                }            }            foreach (var pm in _lstPms)            {                if (pm.IsAvailable)                {                    _dbCallback.PjCreated(pj);                    _dbCallback.PjStart();                    pm.Standby(pj.ProcessRecipe);                    pj.IsStandbyStarted = true;                }            }            return true;        }        #endregion Job task        #region Module task        public Result MonitorModuleTasks()        {            MonitorFIMSTask();            MonitorWaferRobotTask();            MonitorCarrierRobotTask();            MonitorPMTask();            return Result.RUN;        }        private void MonitorWaferRobotTask()        {            if (!_waferRobot.IsAvailable)                return;            foreach (var fims in _lstFIMSs)            {                if (fims.IsWaitTransfer(ModuleName.WaferRobot))                {                    fims.StopWaitTransfer(ModuleName.WaferRobot);                }            }            foreach (var pm in _lstPms)            {                if (pm.IsWaitTransfer(ModuleName.WaferRobot))                {                    pm.StopWaitTransfer(ModuleName.WaferRobot);                }            }            MonitorWaferRobotFIMSTask();            if (!_waferRobot.IsAvailable)                return;            MonitorWaferRobotPMTask();        }        private void MonitorCarrierRobotTask()        {            if (!_carrierRobot.IsAvailable)                return;            //efem robot is idle, release all the target             foreach (var fims in _lstFIMSs)            {                if (fims.IsWaitTransfer(ModuleName.CarrierRobot))                {                    fims.StopWaitTransfer(ModuleName.CarrierRobot);                }            }            foreach (var stock in _lstTotlaStockers)            {                if (stock.IsWaitTransfer(ModuleName.CarrierRobot))                {                    stock.StopWaitTransfer(ModuleName.CarrierRobot);                }            }            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.IsEmptyLayout)                return;            if (pj.ProcessingState == EnumProcessingState.Discharging)            {                //discharge                DischargingOrderBySort(SC.GetStringValue("System.ChargingOrderBySort"));                if (!_carrierRobot.IsAvailable)                    return;                if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                    MonitorCassetteRobotStockerTaskInDischarge(_lstSideDummyStocker, WaferType.SD, _sideDummySlots);                if (!_carrierRobot.IsAvailable)                    return;                MonitorCarrierRobotPickFIMSTask();                if (!_carrierRobot.IsAvailable)                    return;                MonitorCarrierRobotPlaceFIMSTaskInDischarge();            }            else if (pj.ProcessingState == EnumProcessingState.Charging)            {                //charge                if (pj.IsNeedChargeDummyWafer)                    MonitorCassetteRobotStockerTask(_lstSideDummyStocker, WaferType.SD);                if (!_carrierRobot.IsAvailable)                    return;                ChargingOrderBySort(pj, SC.GetStringValue("System.ChargingOrderBySort"));                if (!_carrierRobot.IsAvailable)                    return;                MonitorCarrierRobotPickFIMSTask();                if (!_carrierRobot.IsAvailable)                    return;                MonitorCarrierRobotPlaceFIMSTaskInCharge();            }            else if (pj.ProcessingState == EnumProcessingState.Processing)            {                MonitorCarrierRobotPickFIMSTask();                MonitorCassetteRobotPlaceStockerTask();            }        }        private void MonitorFIMSTask()        {            if (!_lstProcessJobs.Any())                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.IsEmptyLayout)                return;            //load            if (pj.ProcessingState == EnumProcessingState.Charging)            {                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (!CheckCarrierCanPickFromFIMS(st.Module) && !fimsDevice.IsLoadCompleted && st.Load(pj.IsN2PurgeMode, "Charging"))                    {                        return;                    }                }            }            else if (pj.ProcessingState == EnumProcessingState.Discharging)            {                //先判断空片的情况,空片的先load                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    var wafers = WaferManager.Instance.GetWafers(st.Module);                    if (wafers != null)                    {                        if (CheckCarrierNoHasEmptySlot(st.Module) && !fimsDevice.IsLoadCompleted && st.Load(pj.IsN2PurgeMode, "Discharging"))                        {                            return;                        }                    }                }                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    var carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                    if (carrier != null)                    {                        if (!CheckCarrierCanPickFromFIMSInDischarge(carrier.InternalModuleName) && !fimsDevice.IsLoadCompleted && st.Load(pj.IsN2PurgeMode, "Discharging"))                        {                            return;                        }                    }                }            }            //unload            if (pj.ProcessingState == EnumProcessingState.Charging)            {                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (CheckCarrierCanPickFromFIMS(st.Module) && !fimsDevice.IsUnloadCompleted && st.Unload(pj.IsN2PurgeMode, "Charging"))                    {                        return;                    }                }                if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))                {                    SchedulerFIMS fims = null;                    foreach (var st in _lstFIMSs)                    {                        if (!st.IsAvailable || CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                            continue;                        // check FIMS                        var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                        if (fimsDevice.IsUnloadCompleted)                            continue;                        var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);                        var transfer = Singleton<EquipmentManager>.Instance.GetTransferJobStockers();                        if (transfer != null && transfer.Any(x => x.Stocker == carrier.InternalModuleName))                            continue;                        GetWaferSlotsAndWaferType(carrier, out List<int> slots, out WaferType waferType);                        if (slots != null && GetWaferCountRelatedToWaferRobot(waferType, false) <= slots.Count && CheckCarrierCanPickFromStocker(ModuleName.CarrierRobot) && fims.Unload(pj.IsN2PurgeMode, "Charging"))                        {                            return;                        }                    }                }            }            else if (pj.ProcessingState == EnumProcessingState.Discharging)            {                //先判断满片的情况,满片的先unload                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    var carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                    if (carrier != null)                    {                        if (!CheckCarrierHasEmptySlot(st.Module) && _waferRobot.IsAvailable && !fimsDevice.IsUnloadCompleted && st.Unload(pj.IsN2PurgeMode, "Discharging"))                        {                            return;                        }                    }                }                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    var carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                    if (carrier != null)                    {                        if (CheckCarrierCanPickFromFIMSInDischarge(carrier.InternalModuleName) && _waferRobot.IsAvailable && !fimsDevice.IsUnloadCompleted && st.Unload(pj.IsN2PurgeMode, "Discharging"))                        {                            return;                        }                    }                }            }        }        private void MonitorWaferRobotPMTask()        {            if (!_waferRobot.IsAvailable)                return;            var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;            if (pmModule == null)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.IsEmptyLayout)                return;            //place            var wafer = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 2);            List<int> slots = GetWaferSlots(wafer.WaferType);            if (!_logTrigDic.ContainsKey("GetWaferSlots(wafer.WaferType)"))                _logTrigDic.Add("GetWaferSlots(wafer.WaferType)", new R_TRIG());            _logTrigDic["GetWaferSlots(wafer.WaferType)"].CLK = slots == null;            if (_logTrigDic["GetWaferSlots(wafer.WaferType)"].Q)                LOG.Write($"GetWaferSlots(wafer.WaferType) is null");            if (pj.ProcessingState == EnumProcessingState.Charging &&                wafer != null && !wafer.IsEmpty && wafer.ProcessState == EnumWaferProcessStatus.Idle &&                slots != null)            {                Hand placeBlade = Hand.Blade1;                int placeSlot = -1;                for (int i = 0; i < slots.Count; i++)                {                    if (!WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i]))                        continue;                    placeSlot = slots[i];                    placeBlade = Hand.Blade1;                    break;                }                if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 0) &&                    WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 1) &&                    WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 3) &&                    WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 4))                {                    //check PM                    for (int i = 0; i < slots.Count; i++)                    {                        if (!WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i]))                            continue;                        placeSlot = slots[i];                        if (i + 4 < slots.Count && Math.Abs(slots[i] - slots[i + 4]) == 4)                        {                            if (WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 1]) &&                                WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 2]) &&                                WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 3]) &&                                WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 4]))                            {                                placeBlade = Hand.Both;                                if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")                                    placeSlot = slots[i + 4];                            }                        }                        break;                    }                }                if (pmModule.CheckReadyForTransfer(_waferRobot.Module, placeBlade, placeSlot, EnumTransferType.Place, out string reason) &&                    placeSlot != -1 && _waferRobot.Place(_pm1.Module, placeSlot, placeBlade))                {                    _pm1.WaitTransfer(ModuleName.WaferRobot);                    return;                }            }            if (!_waferRobot.IsAvailable)                return;            //pick            List<int> targetWaferSlots = GetWaferSlots();            if (!_logTrigDic.ContainsKey("GetWaferSlots()"))                _logTrigDic.Add("GetWaferSlots()", new R_TRIG());            _logTrigDic["GetWaferSlots()"].CLK = targetWaferSlots == null;            if (_logTrigDic["GetWaferSlots()"].Q)                LOG.Write($"GetWaferSlots() is null");            if (pj.ProcessingState == EnumProcessingState.Discharging &&                wafer != null && wafer.IsEmpty && targetWaferSlots != null)            {                Hand pickBlade = Hand.Blade1;                int pickSlot = -1;                var isEnableDummyWaferDischarge = SC.GetValue<bool>("System.IsEnableDummyWaferDischarge");                //check PM                for (int i = 0; i < targetWaferSlots.Count; i++)                {                    var targetWafer = WaferManager.Instance.GetWafer(_pm1.Module, targetWaferSlots[i]);                    if (targetWafer == null || targetWafer.IsEmpty || (targetWafer.ProcessState != EnumWaferProcessStatus.Completed &&                                                                        targetWafer.ProcessState != EnumWaferProcessStatus.Abort &&                                                                        targetWafer.ProcessState != EnumWaferProcessStatus.Failed))                        continue;                    if (!isEnableDummyWaferDischarge && (targetWafer.WaferType == WaferType.SD || targetWafer.WaferType == WaferType.ED))                        continue;                    pickSlot = targetWaferSlots[i];                    if (i + 4 < targetWaferSlots.Count && Math.Abs(targetWaferSlots[i] - targetWaferSlots[i + 4]) == 4)                    {                        if (WaferManager.Instance.CheckHasWafer(_pm1.Module, targetWaferSlots[i + 1]) &&                            WaferManager.Instance.CheckHasWafer(_pm1.Module, targetWaferSlots[i + 2]) &&                            WaferManager.Instance.CheckHasWafer(_pm1.Module, targetWaferSlots[i + 3]) &&                            WaferManager.Instance.CheckHasWafer(_pm1.Module, targetWaferSlots[i + 4]) &&                            SC.GetValue<bool>($"{_waferRobot.Module}.IsSupport5Wafer"))                        {                            bool isFromTopToBottom = SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom";                            var wafer0 = WaferManager.Instance.GetWafer(_pm1.Module, targetWaferSlots[i + (isFromTopToBottom ? 4 : 0)]);                            var wafer1 = WaferManager.Instance.GetWafer(_pm1.Module, targetWaferSlots[i + (isFromTopToBottom ? 3 : 1)]);                            var wafer2 = WaferManager.Instance.GetWafer(_pm1.Module, targetWaferSlots[i + 2]);                            var wafer3 = WaferManager.Instance.GetWafer(_pm1.Module, targetWaferSlots[i + (isFromTopToBottom ? 1 : 3)]);                            var wafer4 = WaferManager.Instance.GetWafer(_pm1.Module, targetWaferSlots[i + (isFromTopToBottom ? 0 : 4)]);                            if (WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 0) &&                                WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 1) &&                                WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 3) &&                                WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 4) &&                                wafer0.OriginStation == wafer1.OriginStation &&                                wafer0.OriginStation == wafer2.OriginStation &&                                wafer0.OriginStation == wafer3.OriginStation &&                                wafer0.OriginStation == wafer4.OriginStation &&                                Math.Abs(wafer1.OriginSlot - wafer2.OriginSlot) == 1 &&                                Math.Abs(wafer2.OriginSlot - wafer3.OriginSlot) == 1 &&                                Math.Abs(wafer0.OriginSlot - wafer4.OriginSlot) == 4 &&                                Math.Abs(wafer0.OriginSlot - wafer3.OriginSlot) == 3 &&                                Math.Abs(wafer0.OriginSlot - wafer2.OriginSlot) == 2 &&                                Math.Abs(wafer0.OriginSlot - wafer1.OriginSlot) == 1)                            {                                pickBlade = Hand.Both;                                pickSlot = wafer4.Slot;                            }                        }                    }                    break;                }                if (pmModule.CheckReadyForTransfer(_waferRobot.Module, pickBlade, pickSlot, EnumTransferType.Pick, out string reason) &&                    pickSlot != -1 && _waferRobot.Pick(_pm1.Module, pickSlot, pickBlade))                {                    _pm1.WaitTransfer(ModuleName.WaferRobot);                }            }        }        private void MonitorWaferRobotFIMSTask()        {            if (!_waferRobot.IsAvailable)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.IsEmptyLayout)                return;            //pick            if (pj.ProcessingState == EnumProcessingState.Charging &&                WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 2))            {                SchedulerFIMS fims = null;                List<int> slots = null;                WaferType waferType = WaferType.None;                ModuleName currentFims = GetCurrentTransferFims();                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable)                        continue;                    if (!CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    // check FIMS                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (!fimsDevice.IsLoadCompleted)                        continue;                    var carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                    if (carrier != null && !carrier.IsEmpty)                    {                        var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                        bool hasWafer = false;                        var wafers = WaferManager.Instance.GetWafers(st.Module);                        if (wafers != null && stockerInfo != null)                        {                            foreach (var wafer in wafers)                            {                                if (wafer == null || wafer.IsEmpty || wafer.ProcessState != EnumWaferProcessStatus.Idle)                                    continue;                                if (!wafer.IsEmpty && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                                {                                    hasWafer = true;                                    break;                                }                            }                        }                        if (!hasWafer)                            continue;                        if (currentFims != ModuleName.System && currentFims != st.Module)                            continue;                        GetWaferSlotsAndWaferType(carrier, out slots, out waferType);                        if (slots != null && GetWaferCountRelatedToWaferRobot(waferType, false) < slots.Count)                        {                            fims = st;                            break;                        }                    }                }                if (fims != null && slots != null)                {                    Hand pickBlade = Hand.Blade1;                    int pickSlot = -1;                    var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                    var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                    if (WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 0) &&                        WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 1) &&                        WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 3) &&                        WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 4))                    {                        //check PM                        for (int i = 0; i < slots.Count; i++)                        {                            if (!WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i]))                                continue;                            if (i + 4 < slots.Count && Math.Abs(slots[i] - slots[i + 4]) == 4)                            {                                if (WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 1]) &&                                    WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 2]) &&                                    WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 3]) &&                                    WaferManager.Instance.CheckNoWafer(_pm1.Module, slots[i + 4]) &&                                    SC.GetValue<bool>($"{_waferRobot.Module}.IsSupport5Wafer"))                                {                                    pickBlade = Hand.Both;                                    break;                                }                            }                            break;                        }                        //check cassette                        Dictionary<string, int> dummyWafers = new Dictionary<string, int>();                        if (waferType == WaferType.SD)                            dummyWafers = _sideDummyWafers;                        else if (waferType == WaferType.ED)                            dummyWafers = _extraDummyWafers;                        var wafers = WaferManager.Instance.GetWafers(fims.Module).ToList();                        if (SC.GetStringValue("System.ChargingOrderFromCassette") == "FromTopToBottom")                            wafers.Reverse();                        if (wafers != null)                        {                            for (int i = 0; i < wafers.Count; i++)                            {                                if (wafers[i] == null || wafers[i].IsEmpty || wafers[i].ProcessState != EnumWaferProcessStatus.Idle)                                    continue;                                if ((wafers[i].WaferType == WaferType.SD || wafers[i].WaferType == WaferType.ED) && !dummyWafers.ContainsKey(wafers[i].InnerId.ToString()))                                    continue;                                if (stockerInfo.Item3.Any() && !stockerInfo.Item3.Any(x => x == i))                                    continue;                                pickSlot = wafers[i].OriginSlot;                                if (pickBlade == Hand.Both)                                {                                    if (i < SC.GetValue<int>("System.CassetteSlotCount") - 4)                                    {                                        if (waferType == WaferType.SD || waferType == WaferType.ED)                                        {                                            if (!wafers[i + 1].IsEmpty && !wafers[i + 2].IsEmpty && !wafers[i + 3].IsEmpty && !wafers[i + 4].IsEmpty &&                                                wafers[i + 1].ProcessState == EnumWaferProcessStatus.Idle &&                                                wafers[i + 2].ProcessState == EnumWaferProcessStatus.Idle &&                                                wafers[i + 3].ProcessState == EnumWaferProcessStatus.Idle &&                                                wafers[i + 4].ProcessState == EnumWaferProcessStatus.Idle &&                                                (!stockerInfo.Item3.Any() || (stockerInfo.Item3.Any(x => x == i + 1) && stockerInfo.Item3.Any(x => x == i + 2) && stockerInfo.Item3.Any(x => x == i + 3) && stockerInfo.Item3.Any(x => x == i + 4))) &&                                                dummyWafers.ContainsKey(wafers[i + 1].InnerId.ToString()) && dummyWafers.ContainsKey(wafers[i + 2].InnerId.ToString()) &&                                                dummyWafers.ContainsKey(wafers[i + 3].InnerId.ToString()) && dummyWafers.ContainsKey(wafers[i + 4].InnerId.ToString()))                                            {                                                if (SC.GetStringValue("System.ChargingOrderFromCassette") == "FromTopToBottom")                                                    pickSlot = wafers[i + 4].OriginSlot;                                                break;                                            }                                            else                                            {                                                pickBlade = Hand.Blade1;                                                break;                                            }                                        }                                        else                                        {                                            if (!wafers[i + 1].IsEmpty && !wafers[i + 2].IsEmpty && !wafers[i + 3].IsEmpty && !wafers[i + 4].IsEmpty &&                                                wafers[i + 1].ProcessState == EnumWaferProcessStatus.Idle &&                                                wafers[i + 2].ProcessState == EnumWaferProcessStatus.Idle &&                                                wafers[i + 3].ProcessState == EnumWaferProcessStatus.Idle &&                                                wafers[i + 4].ProcessState == EnumWaferProcessStatus.Idle &&                                                (!stockerInfo.Item3.Any() || (stockerInfo.Item3.Any(x => x == i + 1) && stockerInfo.Item3.Any(x => x == i + 2) && stockerInfo.Item3.Any(x => x == i + 3) && stockerInfo.Item3.Any(x => x == i + 4))))                                            {                                                if (SC.GetStringValue("System.ChargingOrderFromCassette") == "FromTopToBottom")                                                    pickSlot = wafers[i + 4].OriginSlot;                                                break;                                            }                                            else                                            {                                                pickBlade = Hand.Blade1;                                                break;                                            }                                        }                                    }                                    else                                    {                                        pickBlade = Hand.Blade1;                                        break;                                    }                                }                                else                                {                                    pickBlade = Hand.Blade1;                                    break;                                }                            }                        }                    }                    if (SC.GetValue<bool>($"System.IsEnableFIMSChargeMapping"))                    {                        if (carrier != null && !carrier.IsEmpty && carrier.IsMapped && carrier.IsMapOK &&                            pickSlot != -1 && _pm1.IsAvailable && _waferRobot.Pick(fims.Module, pickSlot, pickBlade))                        {                            fims.WaitTransfer(ModuleName.WaferRobot);                            return;                        }                    }                    else                    {                        if (pickSlot != -1 && _waferRobot.Pick(fims.Module, pickSlot, pickBlade))                        {                            fims.WaitTransfer(ModuleName.WaferRobot);                        }                    }                }            }            if (!_waferRobot.IsAvailable)                return;            // mapping            if (pj.ProcessingState == EnumProcessingState.Charging && SC.GetValue<bool>($"System.IsEnableFIMSChargeMapping") &&                WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 2) && WaferManager.Instance.CheckNoWafer(ModuleName.WaferRobot, 1))            {                foreach (var fims in _lstFIMSs)                {                    if (!fims.IsAvailable)                        continue;                    if (!CarrierManager.Instance.CheckHasCarrier(fims.Module.ToString(), 0))                        continue;                    // check FIMS                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{fims.Module}");                    if (!fimsDevice.IsLoadCompleted)                        continue;                    var carrier = CarrierManager.Instance.GetCarrier(fims.Module);                    if (carrier == null || carrier.IsEmpty || carrier.IsMapped)                        continue;                    var mappingSlot = GetMappingSlot(fims.Module);                    if (_waferRobot.Map(fims.Module, mappingSlot.Length == 0 ? 0 : Regex.Matches(mappingSlot, "1").Count, mappingSlot, true, false, Hand.Blade1))                    {                        fims.WaitTransfer(ModuleName.WaferRobot);                        return;                    }                }            }            if (!_waferRobot.IsAvailable)                return;            //place            var robotWafer = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 2);            if (pj.ProcessingState == EnumProcessingState.Discharging &&                robotWafer != null && !robotWafer.IsEmpty && (robotWafer.ProcessState == EnumWaferProcessStatus.Completed ||                                                    robotWafer.ProcessState == EnumWaferProcessStatus.Abort ||                                                    robotWafer.ProcessState == EnumWaferProcessStatus.Failed))            {                SchedulerFIMS fims = null;                var waferInRobot = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 2);                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable)                        continue;                    if (!CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    // check FIMS                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (!fimsDevice.IsLoadCompleted)                        continue;                    var carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                    if (carrier == null || carrier.IsEmpty)                        continue;                    if ((ModuleName)waferInRobot.OriginStation != carrier.InternalModuleName)                        continue;                    bool hasEmptySlot = false;                    var wafers = WaferManager.Instance.GetWafers(st.Module);                    if (wafers != null)                    {                        foreach (var wafer in wafers)                        {                            if (wafer == null)                                continue;                            if (wafer.IsEmpty)                            {                                hasEmptySlot = true;                                break;                            }                        }                    }                    if (hasEmptySlot)                    {                        fims = st;                        break;                    }                }                if (fims != null)                {                    Hand placeBlade = Hand.Blade1;                    int placeSlot = waferInRobot.OriginSlot;                    if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 0) &&                        WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 1) &&                        WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 3) &&                        WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 4) &&                        WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 0).OriginStation == WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 1).OriginStation &&                        WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 1).OriginStation == WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 2).OriginStation &&                        WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 2).OriginStation == WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 3).OriginStation &&                        WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 3).OriginStation == WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 4).OriginStation)                    {                        var tempPlaceSlot = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 0).OriginSlot;                        if (tempPlaceSlot + 4 < SC.GetValue<int>("System.CassetteSlotCount"))                        {                            if (WaferManager.Instance.CheckNoWafer(fims.Module, tempPlaceSlot + 1) &&                                WaferManager.Instance.CheckNoWafer(fims.Module, tempPlaceSlot + 2) &&                                WaferManager.Instance.CheckNoWafer(fims.Module, tempPlaceSlot + 3) &&                                WaferManager.Instance.CheckNoWafer(fims.Module, tempPlaceSlot + 4))                            {                                placeBlade = Hand.Both;                                placeSlot = tempPlaceSlot;                            }                        }                    }                    if (placeSlot != -1 && _waferRobot.Place(fims.Module, placeSlot, placeBlade))                    {                        fims.WaitTransfer(ModuleName.WaferRobot);                    }                }            }        }        private bool CheckWaferSlotsHasWafer(List<int> slots)        {            foreach (var slot in slots)            {                if (WaferManager.Instance.CheckHasWafer(_pm1.Module, slot))                {                    return true;                }            }            return false;        }        private List<int> GetWaferSlots()        {            List<int> slots = null;            switch (SC.GetStringValue("System.ChargingOrderBySort"))            {                case "ED,P,M":                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    break;                case "ED,M,P":                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    break;                case "P,ED,M":                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    break;                case "P,M,ED":                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    break;                case "M,P,ED":                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    break;                case "M,ED,P":                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    break;                default:                    if (CheckWaferSlotsHasWafer(_monitor1Slots))                        return _monitor1Slots;                    if (CheckWaferSlotsHasWafer(_monitor2Slots))                        return _monitor2Slots;                    if (CheckWaferSlotsHasWafer(_productSlots))                        return _productSlots;                    if (CheckWaferSlotsHasWafer(_extraDummySlots))                        return _extraDummySlots;                    break;            }            foreach (var slot in _sideDummySlots)            {                if (WaferManager.Instance.CheckHasWafer(_pm1.Module, slot))                {                    return _sideDummySlots;                }            }            return slots;        }        private bool CheckCarrierHasWaferInPM(ModuleName module)        {            bool isHaveWafer = false;            for (int i = 0; i < WaferManager.Instance.GetWafers(_pm1.Module).Length; i++)            {                if (!WaferManager.Instance.CheckHasWafer(_pm1.Module, i))                    continue;                var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                if (wafer == null || wafer.IsEmpty)                    continue;                if (wafer.OriginStation == (int)module)                {                    isHaveWafer = true;                    break;                }            }            return isHaveWafer;        }        private void DischargingOrderBySort(string order)        {            switch (order)            {                case "ED,P,M":                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    if (!_carrierRobot.IsAvailable)                        return;                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    break;                case "ED,M,P":                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    break;                case "P,ED,M":                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    break;                case "P,M,ED":                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    break;                case "M,P,ED":                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    break;                case "M,ED,P":                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    if (!_carrierRobot.IsAvailable)                        return;                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    break;                default:                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M1, _monitor1Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.M2, _monitor2Slots);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.P, _productSlots);                    if (!_carrierRobot.IsAvailable)                        return;                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        MonitorCassetteRobotStockerTaskInDischarge(_lstTotlaStockers, WaferType.ED, _extraDummySlots);                    break;            }        }        private void ChargingOrderBySort(ProcessJobInfo pj, string order)        {            List<SchedulerStocker> pjStocker = new List<SchedulerStocker>();            foreach (var s in pj.Stockers)            {                var module = GetModule(s.Item1) as SchedulerStocker;                if (module != null)                    pjStocker.Add(module);            }            switch (order)            {                case "ED,P,M":                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    break;                case "ED,M,P":                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    break;                case "P,ED,M":                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    if (!_carrierRobot.IsAvailable)                        return;                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    break;                case "P,M,ED":                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    if (!_carrierRobot.IsAvailable)                        return;                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    break;                case "M,P,ED":                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    if (!_carrierRobot.IsAvailable)                        return;                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    break;                case "M,ED,P":                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    if (!_carrierRobot.IsAvailable)                        return;                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    break;                default:                    if (pj.IsNeedChargeDummyWafer)                        MonitorCassetteRobotStockerTask(_lstExtraDummyStocker, WaferType.ED);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.P);                    if (!_carrierRobot.IsAvailable)                        return;                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M1);                    MonitorCassetteRobotStockerTask(pjStocker, WaferType.M2);                    break;            }        }        private void MonitorCassetteRobotStockerTaskInDischarge(List<SchedulerStocker> targetStockers, WaferType targetWaferType, List<int> waferSlots)        {            if (!_carrierRobot.IsAvailable)                return;            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), targetWaferType.ToString()));            //pick            if (CarrierManager.Instance.CheckNoCarrier(ModuleName.CarrierRobot, 0) && waferSlots.Count > 0)            {                bool needPickSdCassette = false;                int sdWaferCount = GetWaferCountRelatedToWaferRobot(targetWaferType, true);                needPickSdCassette = sdWaferCount > 0;                var sdStockerModule = GetModuleNameForNextWafer(targetWaferType);                if (needPickSdCassette && sdStockerModule != ModuleName.System)                {                    SchedulerStocker sdStocker = null;                    foreach (var sd in targetStockers)                    {                        if (!CarrierManager.Instance.CheckHasCarrier(sd.Module, 0) ||                            CarrierManager.Instance.GetCarrier(sd.Module, 0).CarrierType != targetCarrierType)                            continue;                        if (sd.Module == sdStockerModule &&                            CheckCarrierHasEmptySlot(sd.Module))                        {                            sdStocker = sd;                            break;                        }                    }                    SchedulerFIMS fims = null;                    foreach (var st in _lstFIMSs)                    {                        if (!st.IsAvailable)                            continue;                        if (CarrierManager.Instance.CheckNoCarrier(st.Module, 0))                        {                            fims = st;                            break;                        }                    }                    if (sdStocker != null && fims != null)                    {                        if (_carrierRobot.Pick(sdStocker.Module, 0, Hand.Blade1))                        {                            sdStocker.WaitTransfer(ModuleName.CarrierRobot);                            return;                        }                    }                }            }            if (!_carrierRobot.IsAvailable)                return;            //place            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            if (carrier != null && !carrier.IsEmpty && carrier.CarrierType == targetCarrierType)            {                if (!CheckModuleHasWaferInWaferRobotAndPM(carrier.InternalModuleName) || !CheckCarrierHasEmptySlot(ModuleName.CarrierRobot))                {                    SchedulerStocker sdStocker = null;                    foreach (var sd in targetStockers)                    {                        if (CarrierManager.Instance.CheckNoCarrier(sd.Module, 0) && sd.Module == carrier.InternalModuleName)                        {                            sdStocker = sd;                            break;                        }                    }                    if (sdStocker != null)                    {                        var canPlace = true;                        var transfer = Singleton<EquipmentManager>.Instance.GetTransferJobStockers();                        if (transfer != null && transfer.Any(x => x.Stocker == carrier.InternalModuleName))                            canPlace = false;                        if (canPlace && _carrierRobot.Place(sdStocker.Module, 0, Hand.Blade1))                        {                            sdStocker.WaitTransfer(ModuleName.CarrierRobot);                            return;                        }                    }                }            }        }        private bool CheckCarrierCanPickFromFIMSInDischarge(ModuleName fimsCarrierModule)        {            List<ModuleName> robotModules = new List<ModuleName>();            for (int i = 0; i < WaferManager.Instance.GetWafers(ModuleName.WaferRobot).Length; i++)            {                var wafer = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, i);                if (wafer == null || wafer.IsEmpty)                    continue;                if (!robotModules.Contains((ModuleName)wafer.OriginStation))                    robotModules.Add((ModuleName)wafer.OriginStation);            }            if (robotModules.Contains(fimsCarrierModule))                return false;            List<ModuleName> pmModules = new List<ModuleName>();            for (int i = 0; i < WaferManager.Instance.GetWafers(_pm1.Module).Length; i++)            {                var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                if (wafer == null || wafer.IsEmpty)                    continue;                if (!pmModules.Contains((ModuleName)wafer.OriginStation))                    pmModules.Add((ModuleName)wafer.OriginStation);            }            if (pmModules.Contains(fimsCarrierModule))                return false;            return true;        }        private ModuleName GetNextStockerRelatedToWaferRobot()        {            List<ModuleName> fimsModules = new List<ModuleName>();            foreach (var fims in _lstFIMSs)            {                if (CarrierManager.Instance.CheckNoCarrier(fims.Module, 0))                    continue;                fimsModules.Add(CarrierManager.Instance.GetCarrier(fims.Module, 0).InternalModuleName);            }            List<ModuleName> robotModules = new List<ModuleName>();            for (int i = 0; i < WaferManager.Instance.GetWafers(ModuleName.WaferRobot).Length; i++)            {                var wafer = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, i);                if (wafer == null || wafer.IsEmpty)                    continue;                if (!robotModules.Contains((ModuleName)wafer.OriginStation))                    robotModules.Add((ModuleName)wafer.OriginStation);            }            if (robotModules.Count > 0)            {                foreach (var module in robotModules)                {                    if (!fimsModules.Any(x => x == module))                        return module;                }            }            List<ModuleName> pmModules = new List<ModuleName>();            for (int i = 0; i < WaferManager.Instance.GetWafers(_pm1.Module).Length; i++)            {                var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                if (wafer == null || wafer.IsEmpty)                    continue;                if (!pmModules.Contains((ModuleName)wafer.OriginStation))                    pmModules.Add((ModuleName)wafer.OriginStation);                if (pmModules.Count == _lstFIMSs.Count)                    break;            }            if (pmModules.Count > 0)            {                foreach (var module in pmModules)                {                    if (!fimsModules.Any(x => x == module))                        return module;                }            }            return ModuleName.System;        }        private void MonitorCassetteRobotPlaceStockerTask()        {            if (!_carrierRobot.IsAvailable)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return;            //place            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            if (carrier != null && !carrier.IsEmpty && pj.Stockers.Any(x => x.Item1 == carrier.InternalModuleName.ToString()))            {                var target = pj.Stockers.Find(x => x.Item1 == carrier.InternalModuleName.ToString());                if (target != null && CarrierManager.Instance.CheckNoCarrier(target.Item1, 0))                {                    if (_carrierRobot.Place(ModuleHelper.Converter(target.Item1), 0, Hand.Blade1))                    {                        GetModule(target.Item1).WaitTransfer(ModuleName.CarrierRobot);                        return;                    }                }            }        }        private void MonitorCassetteRobotStockerTask(List<SchedulerStocker> targetStockers, WaferType targetWaferType)        {            if (!_carrierRobot.IsAvailable)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.IsEmptyLayout)                return;            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            var targetSlots = GetWaferSlots(targetWaferType);            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), targetWaferType.ToString()));            if (targetSlots == null)                return;            //pick            if (CarrierManager.Instance.CheckNoCarrier(ModuleName.CarrierRobot, 0) && targetSlots.Count > 0)            {                bool needPickTargetCassette = false;                int targetWaferCount = GetWaferCountRelatedToWaferRobot(targetWaferType);                if (targetWaferCount < targetSlots.Count)                    needPickTargetCassette = true;                if (needPickTargetCassette)                {                    SchedulerStocker targetStocker = null;                    foreach (var target in targetStockers)                    {                        if (CarrierManager.Instance.CheckHasCarrier(target.Module, 0) && !CheckCarrierHasNoWafer(target.Module, targetWaferType) &&                            CarrierManager.Instance.GetCarrier(target.Module).CarrierType == targetCarrierType)                        {                            if (pj.ProcessingState == EnumProcessingState.Charging)                            {                                carrier = CarrierManager.Instance.GetCarrier(target.Module);                                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                                var wafers = WaferManager.Instance.GetWafers(target.Module);                                if (wafers != null && stockerInfo != null)                                {                                    foreach (var wafer in wafers)                                    {                                        if (wafer == null || wafer.IsEmpty || wafer.ProcessState != EnumWaferProcessStatus.Idle)                                            continue;                                        if (!wafer.IsEmpty && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                                        {                                            targetStocker = target;                                            break;                                        }                                    }                                    if (targetStocker != null)                                        break;                                }                            }                            else                            {                                targetStocker = target;                                break;                            }                        }                    }                    SchedulerFIMS fims = null;                    foreach (var st in _lstFIMSs)                    {                        if (!st.IsAvailable)                            continue;                        if (CarrierManager.Instance.CheckNoCarrier(st.Module, 0))                        {                            fims = st;                            break;                        }                    }                    if (targetStocker != null && fims != null)                    {                        switch (targetCarrierType.ToString())                        {                            case "SD":                                _lstPlaceSideDummyStocker.Add(targetStocker);                                break;                            case "ED":                                _lstPlaceExtraDummyStockers.Add(targetStocker);                                break;                            case "P":                                _lstPlaceCurrentBatchStocker.Add(targetStocker);                                break;                            case "M1":                                _lstPlaceMonitor1Stocker.Add(targetStocker);                                break;                            case "M2":                                _lstPlaceMonitor2Stocker.Add(targetStocker);                                break;                        }                        if (_carrierRobot.Pick(targetStocker.Module, 0, Hand.Blade1))                        {                            targetStocker.WaitTransfer(ModuleName.CarrierRobot);                            return;                        }                    }                }            }            if (!_carrierRobot.IsAvailable)                return;            //place            carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            if (carrier != null && !carrier.IsEmpty && carrier.CarrierType == targetCarrierType)            {                var needPlace = false;                if (pj.ProcessingState == EnumProcessingState.Charging)                {                    var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                    var wafers = WaferManager.Instance.GetWafers(ModuleName.CarrierRobot);                    if (wafers != null && stockerInfo != null)                    {                        foreach (var wafer in wafers)                        {                            if (wafer == null || wafer.IsEmpty || wafer.ProcessState != EnumWaferProcessStatus.Idle)                                continue;                            if (stockerInfo.Item3.Any() && !stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                            {                                needPlace = true;                                break;                            }                        }                    }                }                if (GetWaferCountRelatedToWaferRobot(targetWaferType) >= targetSlots.Count || CheckCarrierHasNoWafer(ModuleName.CarrierRobot, targetWaferType) || needPlace)                {                    switch (targetCarrierType.ToString())                    {                        case "SD":                            targetStockers = _lstPlaceSideDummyStocker;                            break;                        case "ED":                            targetStockers = _lstPlaceExtraDummyStockers;                            break;                        case "P":                            targetStockers = _lstPlaceCurrentBatchStocker;                            break;                        case "M1":                            targetStockers = _lstPlaceMonitor1Stocker;                            break;                        case "M2":                            targetStockers = _lstPlaceMonitor2Stocker;                            break;                    }                    SchedulerStocker targetStocker = null;                    foreach (var target in targetStockers)                    {                        if (CarrierManager.Instance.CheckNoCarrier(target.Module, 0) && carrier.InternalModuleName == target.Module)                        {                            if (pj.ProcessingState == EnumProcessingState.Charging)                            {                                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                                var wafers = WaferManager.Instance.GetWafers(ModuleName.CarrierRobot);                                if (wafers != null && stockerInfo != null)                                {                                    var hasWafer = false;                                    foreach (var wafer in wafers)                                    {                                        if (wafer == null || wafer.IsEmpty || wafer.ProcessState != EnumWaferProcessStatus.Idle)                                            continue;                                        if (stockerInfo.Item3.Any() && stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                                        {                                            hasWafer = true;                                            break;                                        }                                    }                                    if (!hasWafer)                                    {                                        targetStocker = target;                                        break;                                    }                                }                            }                            else                            {                                targetStocker = target;                                break;                            }                        }                    }                    if (targetStocker != null)                    {                        if (_carrierRobot.Place(targetStocker.Module, 0, Hand.Blade1))                        {                            targetStocker.WaitTransfer(ModuleName.CarrierRobot);                            return;                        }                    }                }            }        }        private bool CheckCarrierCanPickFromStocker(ModuleName currentModule)        {            var currentWafers = WaferManager.Instance.GetWafers(currentModule);            if (currentWafers != null && !currentWafers.Any(x => !x.IsEmpty))            {                //空盒子                return false;            }            var carrierType = CarrierManager.Instance.GetCarrier(currentModule, 0).CarrierType;            var ret = CheckHasPWafer();            var waferTypeSlots = GetWaferTypeSlots();            if (waferTypeSlots != null && waferTypeSlots.Count > 0)            {                Dictionary<WaferType, int> nextWaferTypeDic = new Dictionary<WaferType, int>();                int startIndex = -1;                for (int i = 0; i < waferTypeSlots.Count; i++)                {                    if (WaferManager.Instance.CheckHasWafer(_pm1.Module, waferTypeSlots[i].Item2))                    {                        startIndex = i;                        continue;                    }                    //P wafer不够的情况                    if (waferTypeSlots[i].Item1 == WaferType.P)                    {                        if (!ret.Item1)                            continue;                        else                        {                            if (i - startIndex > ret.Item2)                                continue;                        }                    }                    if (!nextWaferTypeDic.ContainsKey(waferTypeSlots[i].Item1))                        nextWaferTypeDic.Add(waferTypeSlots[i].Item1, 0);                    nextWaferTypeDic[waferTypeSlots[i].Item1] += 1;                    if (nextWaferTypeDic.Count == _lstFIMSs.Count)                        break;                    if (i - startIndex >= _lstFIMSs.Count * 25)                        break;                }                Dictionary<WaferType, int> fimsWaferTypeDic = new Dictionary<WaferType, int>();                foreach (var fims in _lstFIMSs)                {                    if (CarrierManager.Instance.CheckHasCarrier(fims.Module, 0))                    {                        for (int i = 0; i < WaferManager.Instance.GetWafers(fims.Module).Length; i++)                        {                            var wafer = WaferManager.Instance.GetWafer(fims.Module, i);                            if (wafer == null || wafer.IsEmpty)                                continue;                            if (!fimsWaferTypeDic.ContainsKey(wafer.WaferType))                                fimsWaferTypeDic.Add(wafer.WaferType, 0);                            fimsWaferTypeDic[wafer.WaferType] += 1;                        }                    }                }                foreach (var type in nextWaferTypeDic.Keys)                {                    if (carrierType.ToString() == type.ToString())                    {                        if (type == WaferType.P || type == WaferType.M1 || type == WaferType.M2)                        {                            if (!fimsWaferTypeDic.ContainsKey(type) || nextWaferTypeDic[type] > fimsWaferTypeDic[type])                                return true;                        }                        else if (type == WaferType.SD)                        {                            foreach (var wafer in currentWafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (_sideDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    return true;                            }                        }                        else if (type == WaferType.ED)                        {                            foreach (var wafer in currentWafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (_extraDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    return true;                            }                        }                    }                }            }            return false;        }        private bool CheckCarrierCanPickFromFIMS(ModuleName currentModule)        {            var fimsWafers = WaferManager.Instance.GetWafers(currentModule);            if (fimsWafers != null && !fimsWafers.Any(x => !x.IsEmpty))            {                //空盒子                return true;            }            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return true;            var carrier = CarrierManager.Instance.GetCarrier(currentModule, 0);            var carrierType = carrier.CarrierType;            if (!pj.Stockers.Any(x => x.Item1 == carrier.InternalModuleName.ToString()))                return true;            var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());            GetWaferSlotsAndWaferType(carrier, out var slots, out var waferType);            Dictionary<string, int> dummyWafers = new Dictionary<string, int>();            if (waferType == WaferType.SD)                dummyWafers = _sideDummyWafers;            else if (waferType == WaferType.ED)                dummyWafers = _extraDummyWafers;            if (stockerInfo != null)            {                var hasWafer = false;                foreach (var wafer in fimsWafers)                {                    if (wafer == null || wafer.IsEmpty || wafer.ProcessState != EnumWaferProcessStatus.Idle)                        continue;                    if (stockerInfo.Item3.Any() && stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                    {                        hasWafer = true;                        break;                    }                    if (!stockerInfo.Item3.Any())                    {                        if (slots != null && slots.Count > 0)                        {                            for (int i = 0; i < slots.Count; i++)                            {                                if (WaferManager.Instance.CheckHasWafer(_pm1.Module, slots[i]))                                {                                    continue;                                }                                if ((wafer.WaferType == WaferType.SD || wafer.WaferType == WaferType.ED) && !dummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    continue;                                return false;                            }                        }                    }                }                if (hasWafer)                    return false;            }            return true;            var ret = CheckHasPWafer();            var waferTypeSlots = GetWaferTypeSlots();            if (waferTypeSlots != null && waferTypeSlots.Count > 0)            {                List<WaferType> nextWaferTypeLst = new List<WaferType>();                int startIndex = -1;                for (int i = 0; i < waferTypeSlots.Count; i++)                {                    if (WaferManager.Instance.CheckHasWafer(_pm1.Module, waferTypeSlots[i].Item2))                    {                        startIndex = i;                        continue;                    }                    //P wafer不够的情况                    if (waferTypeSlots[i].Item1 == WaferType.P)                    {                        if (!ret.Item1)                            continue;                        else                        {                            if (i - startIndex > ret.Item2)                                continue;                        }                    }                    if (!nextWaferTypeLst.Contains(waferTypeSlots[i].Item1))                        nextWaferTypeLst.Add(waferTypeSlots[i].Item1);                    if (nextWaferTypeLst.Count == _lstFIMSs.Count)                        break;                    if (i - startIndex >= _lstFIMSs.Count * 25)                        break;                }                foreach (var type in nextWaferTypeLst)                {                    if (carrierType.ToString() == type.ToString())                    {                        if (type == WaferType.P || type == WaferType.M1 || type == WaferType.M2)                            return false;                        else if (type == WaferType.SD)                        {                            foreach (var wafer in fimsWafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (_sideDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    return false;                            }                        }                        else if (type == WaferType.ED)                        {                            foreach (var wafer in fimsWafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (_extraDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    return false;                            }                        }                    }                }            }            return true;        }        private bool CheckCarrierCanPlaceToStocker(ModuleName currentModule)        {            var fimsWafers = WaferManager.Instance.GetWafers(currentModule);            if (fimsWafers != null && !fimsWafers.Any(x => !x.IsEmpty))            {                //空盒子                return true;            }            var carrierType = CarrierManager.Instance.GetCarrier(currentModule, 0).CarrierType;            var ret = CheckHasPWafer();            var waferTypeSlots = GetWaferTypeSlots();            if (waferTypeSlots != null && waferTypeSlots.Count > 0)            {                List<WaferType> nextWaferTypeLst = new List<WaferType>();                int startIndex = -1;                for (int i = 0; i < waferTypeSlots.Count; i++)                {                    if (WaferManager.Instance.CheckHasWafer(_pm1.Module, waferTypeSlots[i].Item2))                    {                        startIndex = i;                        continue;                    }                    //P wafer不够的情况                    if (waferTypeSlots[i].Item1 == WaferType.P)                    {                        if (!ret.Item1)                            continue;                        else                        {                            if (i - startIndex > ret.Item2)                                continue;                        }                    }                    if (!nextWaferTypeLst.Contains(waferTypeSlots[i].Item1))                        nextWaferTypeLst.Add(waferTypeSlots[i].Item1);                    if (nextWaferTypeLst.Count == _lstFIMSs.Count)                        break;                    if (i - startIndex >= _lstFIMSs.Count * 25)                        break;                }                foreach (var type in nextWaferTypeLst)                {                    if (carrierType.ToString() == type.ToString())                    {                        if (type == WaferType.P || type == WaferType.M1 || type == WaferType.M2)                            return false;                        else if (type == WaferType.SD)                        {                            foreach (var wafer in fimsWafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (_sideDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    return false;                            }                        }                        else if (type == WaferType.ED)                        {                            foreach (var wafer in fimsWafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (_extraDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    return false;                            }                        }                    }                }            }            return true;        }        private void MonitorCarrierRobotPickFIMSTask()        {            if (!_carrierRobot.IsAvailable)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return;            //pick            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            if (carrier != null && carrier.IsEmpty)            {                SchedulerFIMS fims = null;                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable)                        continue;                    // check FIMS                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (!fimsDevice.IsUnloadCompleted)                        continue;                    if (CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                    {                        carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                        if (carrier != null)                        {                            if (pj.ProcessingState == EnumProcessingState.Discharging)                            {                                //discharge                                if (!CheckCarrierHasEmptySlot(st.Module))                                {                                    fims = st;                                    break;                                }                                if (!CheckModuleHasWaferInWaferRobotAndPM(carrier.InternalModuleName))                                {                                    fims = st;                                    break;                                }                            }                            else if (pj.ProcessingState == EnumProcessingState.Charging)                            {                                //charge                                GetWaferSlotsAndWaferType(carrier, out List<int> slots, out WaferType waferType);                                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                                var wafers = WaferManager.Instance.GetWafers(st.Module);                                if (wafers != null && stockerInfo != null)                                {                                    bool isEmpty = true;                                    foreach (var wafer in wafers)                                    {                                        if (!wafer.IsEmpty)                                        {                                            if (stockerInfo.Item3.Any())                                            {                                                if (stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                                                {                                                    isEmpty = false;                                                    break;                                                }                                            }                                            else                                            {                                                if (waferType == WaferType.P || waferType == WaferType.M1 || waferType == WaferType.M2)                                                {                                                    isEmpty = false;                                                    break;                                                }                                                else if (waferType == WaferType.SD && _sideDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                                {                                                    isEmpty = false;                                                    break;                                                }                                                else if (waferType == WaferType.ED && _extraDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                                {                                                    isEmpty = false;                                                    break;                                                }                                            }                                        }                                    }                                    if (isEmpty)                                    {                                        fims = st;                                        break;                                    }                                }                                if (slots != null && GetWaferCountRelatedToWaferRobot(waferType, false) >= slots.Count)                                {                                    fims = st;                                    break;                                }                            }                            else if (pj.ProcessingState == EnumProcessingState.Processing)                            {                                foreach (var bf in pj.Stockers)                                {                                    if (CarrierManager.Instance.CheckHasCarrier(bf.Item1, 0))                                        continue;                                    if (bf.Item1 == carrier.InternalModuleName.ToString())                                    {                                        fims = st;                                        break;                                    }                                }                            }                        }                    }                }                if (fims != null)                {                    if (_carrierRobot.Pick(fims.Module, 0, Hand.Blade1))                    {                        fims.WaitTransfer(ModuleName.CarrierRobot);                        return;                    }                }            }            if (!_carrierRobot.IsAvailable)                return;        }        private void MonitorCarrierRobotPlaceFIMSTaskInCharge()        {            if (!_carrierRobot.IsAvailable)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return;            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            //place            carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());            if (carrier != null && !carrier.IsEmpty && stockerInfo != null)            {                SchedulerFIMS fims = null;                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable)                        continue;                    // check FIMS                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (!fimsDevice.IsUnloadCompleted)                        continue;                    if (CarrierManager.Instance.CheckNoCarrier(st.Module, 0))                    {                        GetWaferSlotsAndWaferType(carrier, out List<int> slots, out WaferType waferType);                        if (slots != null && GetWaferCountRelatedToWaferRobot(waferType, false) <= slots.Count)                        {                            var wafers = WaferManager.Instance.GetWafers(ModuleName.CarrierRobot);                            foreach (var wafer in wafers)                            {                                if (!wafer.IsEmpty && wafer.WaferType == waferType)                                {                                    if (pj.ProcessingState == EnumProcessingState.Charging && stockerInfo.Item3.Any() && !stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                                        continue;                                    fims = st;                                    break;                                }                            }                        }                        if (fims != null)                            break;                    }                }                if (fims != null)                {                    if (_carrierRobot.Place(fims.Module, 0, Hand.Blade1))                    {                        fims.WaitTransfer(ModuleName.CarrierRobot);                        return;                    }                }            }        }        private void MonitorCarrierRobotPlaceFIMSTaskInDischarge()        {            if (!_carrierRobot.IsAvailable)                return;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return;            var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            //place            carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);            if (carrier != null && !carrier.IsEmpty && CheckCarrierHasEmptySlot(ModuleName.CarrierRobot) && CheckModuleHasWaferInWaferRobotAndPM(carrier.InternalModuleName))            {                SchedulerFIMS fims = null;                foreach (var st in _lstFIMSs)                {                    if (!st.IsAvailable)                        continue;                    // check FIMS                    var fimsDevice = DEVICE.GetDevice<IoFIMS>($"PM1.{st.Module}");                    if (!fimsDevice.IsUnloadCompleted)                        continue;                    if (CarrierManager.Instance.CheckNoCarrier(st.Module, 0))                    {                        fims = st;                        break;                    }                }                if (fims != null)                {                    if (_carrierRobot.Place(fims.Module, 0, Hand.Blade1))                    {                        fims.WaitTransfer(ModuleName.CarrierRobot);                        return;                    }                }            }        }        private bool CheckCarrierHasNoWafer(ModuleName module, WaferType waferType)        {            bool isEmpty = true;            var wafers = WaferManager.Instance.GetWafers(module);            if (wafers != null)            {                foreach (var wafer in wafers)                {                    if (!wafer.IsEmpty)                    {                        if (waferType == WaferType.P || waferType == WaferType.M1 || waferType == WaferType.M2)                        {                            isEmpty = false;                            break;                        }                        else if (waferType == WaferType.SD && _sideDummyWafers.ContainsKey(wafer.InnerId.ToString()))                        {                            isEmpty = false;                            break;                        }                        else if (waferType == WaferType.ED && _extraDummyWafers.ContainsKey(wafer.InnerId.ToString()))                        {                            isEmpty = false;                            break;                        }                    }                }            }            return isEmpty;        }        private Dictionary<string, int> GetDummyWafers(List<SchedulerStocker> targetStockers, int needWaferCount, WaferType waferType)        {            Dictionary<string, int> dummyWafers = new Dictionary<string, int>();            float useCountMin = GetWaferUseCountMin(targetStockers, waferType);            float useCountMax = GetWaferUseCountMax(targetStockers, waferType);            int range = (int)(useCountMax - useCountMin);            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), waferType.ToString()));            for (int i = 0; i <= range; i++)            {                foreach (var target in targetStockers)                {                    var carrier = CarrierManager.Instance.GetCarrier(target.Module);                    if (carrier == null || carrier.IsEmpty || carrier.CarrierType != targetCarrierType)                        continue;                    var wafers = WaferManager.Instance.GetWafers(target.Module);                    if (wafers != null)                    {                        foreach (var wafer in wafers)                        {                            if (wafer.IsEmpty)                                continue;                            if (wafer.UseCount <= useCountMin && dummyWafers.Count() < needWaferCount)                            {                                if (!dummyWafers.ContainsKey(wafer.InnerId.ToString()))                                    dummyWafers.Add(wafer.InnerId.ToString(), wafer.OriginStation);                            }                        }                    }                }                if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))                {                    var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);                    if (carrier.CarrierType == targetCarrierType)                    {                        var wafers = WaferManager.Instance.GetWafers(ModuleName.CarrierRobot);                        if (wafers != null)                        {                            foreach (var wafer in wafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (wafer.UseCount <= useCountMin && dummyWafers.Count() < needWaferCount)                                {                                    if (!dummyWafers.ContainsKey(wafer.InnerId.ToString()))                                        dummyWafers.Add(wafer.InnerId.ToString(), wafer.OriginStation);                                }                            }                        }                    }                }                foreach (var fims in _lstFIMSs)                {                    if (!fims.IsOnline)                        continue;                    var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                    if (!carrier.IsEmpty && carrier.CarrierType == targetCarrierType)                    {                        var wafers = WaferManager.Instance.GetWafers(fims.Module);                        if (wafers != null)                        {                            foreach (var wafer in wafers)                            {                                if (wafer.IsEmpty)                                    continue;                                if (wafer.UseCount <= useCountMin && dummyWafers.Count() < needWaferCount)                                {                                    if (!dummyWafers.ContainsKey(wafer.InnerId.ToString()))                                        dummyWafers.Add(wafer.InnerId.ToString(), wafer.OriginStation);                                }                            }                        }                    }                }                if (dummyWafers.Count() < needWaferCount)                    useCountMin++;                else                    break;            }            return dummyWafers;        }        private float GetWaferUseCountMin(List<SchedulerStocker> targetStockers, WaferType waferType)        {            float iUseCountMin = 10000;            foreach (var target in targetStockers)            {                var wafers = WaferManager.Instance.GetWafers(target.Module);                if (wafers == null)                    continue;                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty || wafer.WaferType != waferType)                        continue;                    if (wafer.UseCount < iUseCountMin)                        iUseCountMin = wafer.UseCount;                }            }            if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))            {                var wafers = WaferManager.Instance.GetWafers(ModuleName.CarrierRobot);                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty || wafer.WaferType != waferType)                        continue;                    if (wafer.UseCount < iUseCountMin)                        iUseCountMin = wafer.UseCount;                }            }            foreach (var fims in _lstFIMSs)            {                if (!fims.IsOnline)                    continue;                var wafers = WaferManager.Instance.GetWafers(fims.Module);                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty || wafer.WaferType != waferType)                        continue;                    if (wafer.UseCount < iUseCountMin)                        iUseCountMin = wafer.UseCount;                }            }            return iUseCountMin;        }        private float GetWaferUseCountMax(List<SchedulerStocker> targetStockers, WaferType waferType)        {            float iUseCountMax = 0;            foreach (var target in targetStockers)            {                var wafers = WaferManager.Instance.GetWafers(target.Module);                if (wafers == null)                    continue;                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty || wafer.WaferType != waferType)                        continue;                    if (wafer.UseCount >= iUseCountMax)                        iUseCountMax = wafer.UseCount;                }            }            if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))            {                var wafers = WaferManager.Instance.GetWafers(ModuleName.CarrierRobot);                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty || wafer.WaferType != waferType)                        continue;                    if (wafer.UseCount >= iUseCountMax)                        iUseCountMax = wafer.UseCount;                }            }            foreach (var fims in _lstFIMSs)            {                if (!fims.IsOnline)                    continue;                var wafers = WaferManager.Instance.GetWafers(fims.Module);                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty || wafer.WaferType != waferType)                        continue;                    if (wafer.UseCount >= iUseCountMax)                        iUseCountMax = wafer.UseCount;                }            }            return iUseCountMax;        }        private ModuleName GetModuleNameForNextWafer(WaferType waferType)        {            ModuleName module = ModuleName.System;            bool hasEmptyFims = false;            foreach (var fims in _lstFIMSs)            {                if (CarrierManager.Instance.CheckNoCarrier(fims.Module, 0))                {                    hasEmptyFims = true;                    break;                }            }            if (!hasEmptyFims)                return module;            if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 0))            {                module = (ModuleName)WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 0).OriginStation;            }            else if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 1))            {                module = (ModuleName)WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 1).OriginStation;            }            else if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 2))            {                module = (ModuleName)WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 2).OriginStation;            }            else if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 3))            {                module = (ModuleName)WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 3).OriginStation;            }            else if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 4))            {                module = (ModuleName)WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 4).OriginStation;            }            else            {                if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")                {                    for (int i = 0; i < SC.GetValue<int>($"Boat.SlotCount"); i++)                    {                        if (!WaferManager.Instance.CheckHasWafer(_pm1.Module, i))                            continue;                        var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                        if (wafer == null || wafer.IsEmpty)                            continue;                        if (wafer.WaferType == waferType)                        {                            module = (ModuleName)WaferManager.Instance.GetWafer(_pm1.Module, i).OriginStation;                            break;                        }                    }                }                else                {                    for (int i = SC.GetValue<int>($"Boat.SlotCount") - 1; i >= 0; i--)                    {                        if (!WaferManager.Instance.CheckHasWafer(_pm1.Module, i))                            continue;                        var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                        if (wafer == null || wafer.IsEmpty)                            continue;                        if (wafer.WaferType == waferType)                        {                            module = (ModuleName)WaferManager.Instance.GetWafer(_pm1.Module, i).OriginStation;                            break;                        }                    }                }            }            //下一片要放回的cassette已经在fims上            bool isCarrierAlreadyPicked = false;            foreach (var fims in _lstFIMSs)            {                if (!CarrierManager.Instance.CheckHasCarrier(fims.Module, 0))                    continue;                var carrier = CarrierManager.Instance.GetCarrier(fims.Module);                if (carrier != null && carrier.InternalModuleName == module)                {                    isCarrierAlreadyPicked = true;                }            }            if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))            {                var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot);                if (carrier != null && carrier.InternalModuleName == module)                {                    isCarrierAlreadyPicked = true;                }            }            if (isCarrierAlreadyPicked)            {                if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")                {                    for (int i = 0; i < SC.GetValue<int>($"Boat.SlotCount"); i++)                    {                        if (!WaferManager.Instance.CheckHasWafer(_pm1.Module, i))                            continue;                        var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                        if (wafer == null || wafer.IsEmpty)                            continue;                        if (wafer.WaferType == waferType && (ModuleName)WaferManager.Instance.GetWafer(_pm1.Module, i).OriginStation != module)                        {                            module = (ModuleName)WaferManager.Instance.GetWafer(_pm1.Module, i).OriginStation;                            break;                        }                    }                }                else                {                    for (int i = SC.GetValue<int>($"Boat.SlotCount") - 1; i >= 0; i--)                    {                        if (!WaferManager.Instance.CheckHasWafer(_pm1.Module, i))                            continue;                        var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                        if (wafer == null || wafer.IsEmpty)                            continue;                        if (wafer.WaferType == waferType && (ModuleName)WaferManager.Instance.GetWafer(_pm1.Module, i).OriginStation != module)                        {                            module = (ModuleName)WaferManager.Instance.GetWafer(_pm1.Module, i).OriginStation;                            break;                        }                    }                }            }            return module;        }        private int GetWaferCountRelatedToWaferRobot(WaferType waferType, bool includeFims = true)        {            int count = 0;            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.IsEmptyLayout)                return count;            if (includeFims)            {                foreach (var fims in _lstFIMSs)                {                    if (CarrierManager.Instance.CheckNoCarrier(fims.Module, 0))                        continue;                    var wafers = WaferManager.Instance.GetWafers(fims.Module);                    var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                    var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                    foreach (var wafer in wafers)                    {                        if (!wafer.IsEmpty && wafer.WaferType == waferType)                        {                            if (pj.ProcessingState == EnumProcessingState.Charging && stockerInfo != null && stockerInfo.Item3.Any() && !stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                                continue;                            if (waferType == WaferType.P || waferType == WaferType.M1 || waferType == WaferType.M2)                                count++;                            else if (waferType == WaferType.SD && _sideDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                count++;                            else if (waferType == WaferType.ED && _extraDummyWafers.ContainsKey(wafer.InnerId.ToString()))                                count++;                        }                    }                }            }            for (int i = 0; i < WaferManager.Instance.GetWafers(ModuleName.WaferRobot).Length; i++)            {                var wafer = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, i);                if (!wafer.IsEmpty && wafer.WaferType == waferType)                {                    var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == ((ModuleName)wafer.OriginStation).ToString());                    if (pj.ProcessingState == EnumProcessingState.Charging && stockerInfo != null && stockerInfo.Item3.Any() && !stockerInfo.Item3.Any(x => x == wafer.OriginSlot))                        continue;                    count++;                }            }            for (int i = 0; i < WaferManager.Instance.GetWafers(_pm1.Module).Length; i++)            {                var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                if (!wafer.IsEmpty && wafer.WaferType == waferType)                    count++;            }            return count;        }        private void GetWaferSlotsAndWaferType(CarrierInfo carrier, out List<int> slots, out WaferType waferType)        {            slots = null;            waferType = WaferType.None;            if (carrier != null && !carrier.IsEmpty)            {                switch (carrier.CarrierType)                {                    case CarrierType.SD:                        slots = _sideDummySlots;                        waferType = WaferType.SD;                        break;                    case CarrierType.ED:                        slots = _extraDummySlots;                        waferType = WaferType.ED;                        break;                    case CarrierType.M1:                        slots = _monitor1Slots;                        waferType = WaferType.M1;                        break;                    case CarrierType.M2:                        slots = _monitor2Slots;                        waferType = WaferType.M2;                        break;                    case CarrierType.P:                        slots = _productSlots;                        waferType = WaferType.P;                        break;                }            }        }        private bool CheckCarrierHasEmptySlot(ModuleName module)        {            bool hasEmpty = false;            var wafers = WaferManager.Instance.GetWafers(module);            if (wafers != null)            {                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty)                    {                        hasEmpty = true;                        break;                    }                }            }            return hasEmpty;        }        private bool CheckCarrierNoHasEmptySlot(ModuleName module)        {            bool noHasEmpty = true;            var wafers = WaferManager.Instance.GetWafers(module);            if (wafers != null)            {                foreach (var wafer in wafers)                {                    if (!wafer.IsEmpty)                    {                        noHasEmpty = false;                        break;                    }                }            }            return noHasEmpty;        }        private bool CheckModuleHasWaferInWaferRobotAndPM(ModuleName module)        {            bool isHaveWafer = false;            var wafers = WaferManager.Instance.GetWafers(ModuleName.PM1).ToList();            if (wafers != null)            {                foreach (var wafer in wafers)                {                    if (wafer == null || wafer.IsEmpty)                        continue;                    if (wafer.OriginStation == (int)module)                    {                        isHaveWafer = true;                        break;                    }                }            }            wafers = WaferManager.Instance.GetWafers(ModuleName.WaferRobot).ToList();            if (wafers != null)            {                foreach (var wafer in wafers)                {                    if (wafer == null || wafer.IsEmpty)                        continue;                    if (wafer.OriginStation == (int)module)                    {                        isHaveWafer = true;                        break;                    }                }            }            return isHaveWafer;        }        private bool CheckWaferStatus(List<Tuple<string, string, List<int>>> useStockers, bool isNeedCheckMonitor, bool isNeedCheckP, out string reason)        {            reason = string.Empty;            if (SC.GetValue<bool>("System.IsCycleMode"))            {                foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)                {                    if (!ModuleHelper.IsStocker(module) || !useStockers.Any(x => x.Item1 == module.ToString()))                        continue;                    var wafers = WaferManager.Instance.GetWafers(module);                    if (wafers == null)                        continue;                    foreach (var wafer in wafers)                    {                        if (wafer.IsEmpty)                            continue;                        wafer.ProcessState = EnumWaferProcessStatus.Idle;                    }                }            }            else            {                foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)                {                    if (!ModuleHelper.IsStocker(module) || !useStockers.Any(x => x.Item1 == module.ToString()))                        continue;                    var stockerInfo = useStockers.FirstOrDefault(x => x.Item1 == module.ToString());                    var wafers = WaferManager.Instance.GetWafers(module);                    if (wafers == null || stockerInfo == null)                        continue;                    if (SC.GetValue<bool>("System.EnableWarningFailedWafer"))                    {                        if (wafers.ToList().Any(x => x.ProcessState == EnumWaferProcessStatus.Failed))                        {                            reason = $"{module} has failed wafers.";                            return false;                        }                    }                    foreach (var wafer in wafers)                    {                        if (wafer.IsEmpty)                            continue;                        if (wafer.ProcessState != EnumWaferProcessStatus.Idle && SC.GetValue<bool>("System.EnableWarningWaferProcessState") && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                        {                            if (wafer.WaferType == WaferType.P && isNeedCheckP)                            {                                reason = $"{(ModuleName)wafer.OriginStation} {wafer.WaferType} wafer already completed process.";                                return false;                            }                            if (isNeedCheckMonitor)                            {                                if (wafer.WaferType == WaferType.M2 || wafer.WaferType == WaferType.M || wafer.WaferType == WaferType.M1)                                {                                    reason = $"{(ModuleName)wafer.OriginStation} {wafer.WaferType} wafer already completed process.";                                    return false;                                }                            }                        }                        if (wafer.WaferType == WaferType.ED || wafer.WaferType == WaferType.SD)                            wafer.ProcessState = EnumWaferProcessStatus.Idle;                    }                }            }            return true;        }        private Tuple<bool, int> CheckHasPWafer()        {            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null || pj.Stockers == null)                return Tuple.Create(false, 0);            int countP = 0;            foreach (var stocker in pj.Stockers)            {                var wafers = WaferManager.Instance.GetWafersByOriginalPosition(ModuleHelper.Converter(stocker.Item1));                foreach (var wafer in wafers)                {                    if (!wafer.IsEmpty && wafer.WaferType == WaferType.P && wafer.Station != (int)ModuleName.PM1)                    {                        countP++;                    }                }            }            return Tuple.Create(countP > 0, countP);        }        //返回下一片要放的wafer的slot和type        private int GetNextWaferSlot(out WaferType waferType)        {            var waferTypeSlots = GetWaferTypeSlots();            waferType = WaferType.None;            var ret = CheckHasPWafer();            int slot = -1;            if (waferTypeSlots != null)            {                for (int i = 0; i < waferTypeSlots.Count; i++)                {                    if (WaferManager.Instance.CheckHasWafer(ModuleName.PM1, waferTypeSlots[i].Item2))                    {                        continue;                    }                    //针对P wafer不够的情况                    if (!ret.Item1 && waferTypeSlots[i].Item1 == WaferType.P)                        continue;                    slot = waferTypeSlots[i].Item2;                    waferType = waferTypeSlots[i].Item1;                    break;                }            }            return slot;        }        //返回下一片要放的wafer的type,需要除去当前在机械手上的wafer        private WaferType GetNextWaferType()        {            var ret = CheckHasPWafer();            var waferTypeSlots = GetWaferTypeSlots();            WaferType waferType = WaferType.None;            if (waferTypeSlots != null)            {                for (int i = 0; i < waferTypeSlots.Count; i++)                {                    if (WaferManager.Instance.CheckHasWafer(ModuleName.PM1, waferTypeSlots[i].Item2))                    {                        continue;                    }                    var wafer0 = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 0);                    var wafer1 = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 1);                    if (i < waferTypeSlots.Count && wafer0 != null && !wafer0.IsEmpty && wafer0.WaferType == waferTypeSlots[i].Item1)                        i++;                    if (i < waferTypeSlots.Count && wafer1 != null && !wafer1.IsEmpty && wafer1.WaferType == waferTypeSlots[i].Item1)                        i++;                    //机械手wafer放过去就放满了                    if (i >= waferTypeSlots.Count)                    {                        break;                    }                    //针对P wafer不够的情况                    if (!ret.Item1 && waferTypeSlots[i].Item1 == WaferType.P)                        continue;                    waferType = waferTypeSlots[i].Item1;                    break;                }            }            return waferType;        }        private ModuleName GetCurrentTransferFims()        {            ModuleName currentFims = ModuleName.System;            foreach (var st in _lstFIMSs)            {                if (!st.IsAvailable)                    continue;                if (!CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                    continue;                var carrier = CarrierManager.Instance.GetCarrier(st.Module, 0);                if (carrier != null && !carrier.IsEmpty)                {                    bool hasWafer = false;                    var wafers = WaferManager.Instance.GetWafers(st.Module);                    if (wafers != null)                    {                        foreach (var wafer in wafers)                        {                            if (wafer == null || wafer.IsEmpty)                                continue;                            if (!wafer.IsEmpty)                            {                                hasWafer = true;                                break;                            }                        }                    }                    if (hasWafer && CheckCarrierHasWaferInPM(carrier.InternalModuleName))                    {                        currentFims = st.Module;                        break;                    };                }            }            return currentFims;        }        private List<int> GetWaferSlots(out WaferType waferType)        {            var waferTypeSlots = GetWaferTypeSlots();            waferType = WaferType.None;            var slots = new List<int>();            if (waferTypeSlots != null)            {                List<Tuple<ModuleName, WaferType, int>> fimsWaferInforLst = new List<Tuple<ModuleName, WaferType, int>>();                foreach (var st in _lstFIMSs)                {                    if (!CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                        continue;                    var fimsWafers = WaferManager.Instance.GetWafers(st.Module);                    int waferCountFIMS = 0;                    WaferType waferTypeFIMS = WaferType.None;                    if (fimsWafers != null)                    {                        foreach (var fimsWafer in fimsWafers)                        {                            if (!fimsWafer.IsEmpty)                            {                                waferTypeFIMS = fimsWafer.WaferType;                                if (fimsWafer.WaferType == WaferType.P || fimsWafer.WaferType == WaferType.M1 || fimsWafer.WaferType == WaferType.M2)                                {                                    waferCountFIMS++;                                }                                else if (fimsWafer.WaferType == WaferType.SD && _sideDummyWafers.ContainsKey(fimsWafer.InnerId.ToString()))                                {                                    waferCountFIMS++;                                }                                else if (fimsWafer.WaferType == WaferType.ED && _extraDummyWafers.ContainsKey(fimsWafer.InnerId.ToString()))                                {                                    waferCountFIMS++;                                }                            }                        }                    }                    if (waferCountFIMS > 0)                        fimsWaferInforLst.Add(Tuple.Create(st.Module, waferTypeFIMS, waferCountFIMS));                }                //两个FIMS都有Carrier,直接返回                if (_lstFIMSs.Count == fimsWaferInforLst.Count)                    return slots;                //FIMS有空的,才继续                int startIndex = 0;                for (int i = 0; i < waferTypeSlots.Count; i++)                {                    if (WaferManager.Instance.CheckHasWafer(ModuleName.PM1, waferTypeSlots[i].Item2))                    {                        continue;                    }                    var wafer0 = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 0);                    var wafer1 = WaferManager.Instance.GetWafer(ModuleName.WaferRobot, 1);                    if (i < waferTypeSlots.Count && wafer0 != null && !wafer0.IsEmpty && wafer0.WaferType == waferTypeSlots[i].Item1)                        i++;                    if (i >= waferTypeSlots.Count)                        break;                    if (i < waferTypeSlots.Count && wafer1 != null && !wafer1.IsEmpty && wafer1.WaferType == waferTypeSlots[i].Item1)                        i++;                    if (i >= waferTypeSlots.Count)                        break;                    if (fimsWaferInforLst.Count > 0)                    {                        var fimsWaferInfor = fimsWaferInforLst.FirstOrDefault(x => x.Item2 == waferTypeSlots[i].Item1);                        if (fimsWaferInfor != null)                        {                            for (; i < waferTypeSlots.Count;)                            {                                if (i >= waferTypeSlots.Count)                                    break;                                if (waferTypeSlots[i].Item1 == fimsWaferInfor.Item2)                                    i++;                                else                                    break;                            }                        }                    }                    if (i >= waferTypeSlots.Count)                    {                        i = waferTypeSlots.Count - 1;                    }                    startIndex = i;                    waferType = waferTypeSlots[i].Item1;                    break;                }                for (int i = startIndex; i < waferTypeSlots.Count; i++)                {                    if (waferTypeSlots[i].Item1 != waferType)                    {                        break;                    }                    slots.Add(waferTypeSlots[i].Item2);                }            }            return slots;        }        private List<int> GetWaferSlots(WaferType waferType)        {            var slots = new List<int>();            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return null;            if (!RecipeParser.LayoutRecipeParse(pj.LayoutRecipe, ModuleName.PM1.ToString(), out RecipeLayoutEntityNormal layoutRecipeDataNormal, out RecipeLayoutEntityExpert layoutRecipeDataExpert, out string reason))            {                //Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Load layout recipe {pj.LayoutRecipe} failed, {reason}");                return null;            }            switch (waferType)            {                case WaferType.SD:                    return pj.SideDummySlots;                case WaferType.ED:                    return pj.ExtraDummySlots;                case WaferType.M1:                    return pj.Monitor1Slots;                case WaferType.M2:                    return pj.Monitor2Slots;                case WaferType.P:                    return pj.ProductSlots;            }            return null;        }        private List<int> GetWaferSlots(RecipeLayoutEntityExpert layoutRecipeDataExpert, WaferType waferType)        {            var slots = new List<int>();            if (layoutRecipeDataExpert == null)                return null;            var type = "";            switch (waferType)            {                case WaferType.SD:                    type = "sd";                    break;                case WaferType.ED:                    type = "fd";                    break;                case WaferType.M1:                    type = "m1";                    break;                case WaferType.M2:                    type = "m2";                    break;                case WaferType.P:                    type = "pd";                    break;            }            if (layoutRecipeDataExpert != null)            {                for (int i = 0; i < layoutRecipeDataExpert.Items.Count; i++)                {                    if (layoutRecipeDataExpert.Items[i].ToLower() == type)                    {                        slots.Add(i);                    }                }                if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")                {                    slots.Sort((x, y) => -x.CompareTo(y));//降序                }                else                {                    slots.Sort((x, y) => x.CompareTo(y));//升序                }            }            return slots;        }        #region charge rule        private bool CheckChargeRule(ProcessJobInfo pj, out bool waferShort, bool needCheckPWafer, bool needCheckMWafer, out string reason)        {            waferShort = false;            //needCheckPWafer = false;            reason = string.Empty;            if (!RecipeParser.LayoutRecipeParse(pj.LayoutRecipe, ModuleName.PM1.ToString(), out RecipeLayoutEntityNormal layoutRecipeDataNormal, out RecipeLayoutEntityExpert layoutRecipeDataExpert, out reason))            {                //Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Load layout recipe {pj.LayoutRecipe} failed, {reason}");                return false;            }            if (!CheckLayoutParameter(layoutRecipeDataNormal, out reason))            {                return false;            }            int iCassetteSlotCount = SC.GetValue<int>("System.CassetteSlotCount");            if (iCassetteSlotCount == 0)            {                reason = "System.CassetteSlotCount not valid";                return false;            }            int iSlotCount = SC.GetValue<int>($"Boat.SlotCount");            var productSlots = GetWaferSlots(layoutRecipeDataExpert, WaferType.P);            var monitor1Slots = GetWaferSlots(layoutRecipeDataExpert, WaferType.M1);            var monitor2Slots = GetWaferSlots(layoutRecipeDataExpert, WaferType.M2);            var sideDummySlots = GetWaferSlots(layoutRecipeDataExpert, WaferType.SD);            var extraDummySlots = GetWaferSlots(layoutRecipeDataExpert, WaferType.ED);            int iNeedCassetteNum;            var temp = (float)productSlots.Count / iCassetteSlotCount;            iNeedCassetteNum = (int)Math.Ceiling(temp);            int iCurrentCassetteNum = GetCurrentCassetteNum(pj, WaferType.P);            int iDummyUpper = int.Parse(layoutRecipeDataNormal.DummyUpperSlot);            int iDummyLower = int.Parse(layoutRecipeDataNormal.DummyLowerSlot);            int iNeedPWaferNum = productSlots.Count;            int iNeedM1WaferNum = monitor1Slots.Count;            int iNeedM2WaferNum = monitor2Slots.Count;            int iNeedSDWaferNum = sideDummySlots.Count;            int iNeedEDWaferNum = extraDummySlots.Count;            int iCurrentPWaferNum = GetCurrentWaferNum(pj, WaferType.P, out bool pSlotSelect);            int iCurrentSDWaferNum = GetCurrentDummyWaferNum(WaferType.SD);            int iCurrentM1WaferNum = GetCurrentWaferNum(pj, WaferType.M1, out bool m1SlotSelect);            int iCurrentM2WaferNum = GetCurrentWaferNum(pj, WaferType.M2, out bool m2SlotSelect);            int iCurrentEDWaferNum = GetCurrentDummyWaferNum(WaferType.ED);            //if (iNeedMWaferNum > iCurrentMWaferNum)            //{            //    reason = "monitor wafer are short.";            //    return false;            //}            //if (iNeedSDWaferNum > iCurrentSDWaferNum)            //{            //    reason = "dummy wafer are short.";            //    return false;            //}            //if (iNeedEDWaferNum > iCurrentEDWaferNum)            //{            //    reason = "extra dummy wafer are short.";            //    return false;            //}            if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")            {                sideDummySlots.Sort((x, y) => -x.CompareTo(y));//降序                extraDummySlots.Sort((x, y) => -x.CompareTo(y));//降序                monitor1Slots.Sort((x, y) => -x.CompareTo(y));//降序                monitor2Slots.Sort((x, y) => -x.CompareTo(y));//降序                productSlots.Sort((x, y) => -x.CompareTo(y));//降序            }            else            {                sideDummySlots.Sort((x, y) => x.CompareTo(y));//升序                extraDummySlots.Sort((x, y) => x.CompareTo(y));//升序                monitor1Slots.Sort((x, y) => x.CompareTo(y));//升序                monitor2Slots.Sort((x, y) => x.CompareTo(y));//升序                productSlots.Sort((x, y) => x.CompareTo(y));//升序            }            pj.SideDummySlots = sideDummySlots;            pj.ExtraDummySlots = extraDummySlots;            pj.Monitor1Slots = monitor1Slots;            pj.Monitor2Slots = monitor2Slots;            pj.ProductSlots = productSlots;            bool bM1Short = false;            bool bM2Short = false;            if (needCheckMWafer)            {                bM1Short = GetRemainWaferNum(pj, WaferType.M1) < 0 && monitor1Slots.Count > 0;                bM2Short = GetRemainWaferNum(pj, WaferType.M2) < 0 && monitor2Slots.Count > 0;                if (monitor1Slots.Count > 0 && m1SlotSelect && iCurrentM1WaferNum != monitor1Slots.Count)                {                    reason = $"layout recipe M1 wafer number={iCurrentM1WaferNum},  current M1 wafer number={monitor1Slots.Count}, not match";                    return false;                }                if (monitor2Slots.Count > 0 && m2SlotSelect && iCurrentM2WaferNum != monitor2Slots.Count)                {                    reason = $"layout recipe M2 wafer number={iCurrentM2WaferNum},  current M2 wafer number={monitor2Slots.Count}, not match";                    return false;                }                if (bM1Short)                {                    reason = "M1 wafer are short";                    return false;                }                if (bM2Short)                {                    reason = "M2 wafer are short";                    return false;                }            }            if (!needCheckPWafer)                return true;            if (iNeedPWaferNum > iCurrentPWaferNum)            {                reason = "layout recipe production wafer number greater than current production wafer number";                return false;            }            if (pSlotSelect && iNeedPWaferNum != iCurrentPWaferNum)            {                reason = $"layout recipe production wafer number={iNeedPWaferNum},  current production wafer number={iCurrentPWaferNum}, not match";                return false;            }            //下面charge rule暂时不判断            return true;            //if (layoutRecipeDataNormal.ProductPosition.ToLower() == "slot")            //{            //    if (iNeedPWaferNum <= iCurrentPWaferNum)            //    {            //        return true;            //    }            //    else            //    {            //        reason = "production wafer insufficiency productPos set to slot";            //        return false;            //    }            //}            string strWhenCassetteAreShort = layoutRecipeDataNormal.WhenCassetteInBatchAreShort;            string strWhenPWaferShort = layoutRecipeDataNormal.WhenPWaferShort;            string strWhenWaferInCassetteAreShort = layoutRecipeDataNormal.WhenWaferInCassetteAreShort;            string strWhenEDAreShort = layoutRecipeDataNormal.WhenEDAreShort;            bool bSDMove = layoutRecipeDataNormal.SDRule == "Move";            bool bBoatSupplyED = layoutRecipeDataNormal.RuleOfSpaceInBoat == "SupplyED" && !bSDMove;            bool bEDShort = false;            List<string> adjustedSlot = new List<string>();            for (int i = 0; i < layoutRecipeDataExpert.Items.Count; i++)            {                adjustedSlot.Add(layoutRecipeDataExpert.Items[i]);            }            bool bCassetteSuppleED;            bool bWaferSuppleED;            if (iNeedPWaferNum > iCurrentPWaferNum)            {                // 先判断cassette是否充足,然后再判断P wafers是否充足                if (iNeedCassetteNum > iCurrentCassetteNum)                {                    if (strWhenCassetteAreShort != "LeaveAs" && strWhenCassetteAreShort != "SupplyED")                    {                        reason = "cassette in batch are short";                        return true;                    }                    else                    {                        bCassetteSuppleED = strWhenCassetteAreShort == "SupplyED";                        // 重新计算需要的P wafers                        iNeedPWaferNum = iNeedPWaferNum - (iNeedCassetteNum - iCurrentCassetteNum) * iCassetteSlotCount;                        // 判断P wafers是否充足                        if (iNeedPWaferNum > iCurrentPWaferNum)                        {   // 只有在when P wafer short == leave as 的时候 when wafer in cassette are short 才执行判断                            if (strWhenPWaferShort == "LeaveAs")                            {                                if (strWhenWaferInCassetteAreShort != "LeaveAs" && strWhenWaferInCassetteAreShort != "SupplyED")                                {                                    reason = "wafer in cassette are short";                                    return true;                                }                                else                                {                                    bWaferSuppleED = strWhenWaferInCassetteAreShort == "SupplyED";                                    bEDShort = CassetteShortWaferInCassetteShort(pj, ref adjustedSlot, ref iCurrentEDWaferNum, layoutRecipeDataNormal, layoutRecipeDataExpert, bCassetteSuppleED, bWaferSuppleED);                                }                            }                            else                            {                                bEDShort = CassetteShortPWaferShort(pj, ref adjustedSlot, ref iCurrentEDWaferNum, layoutRecipeDataNormal, layoutRecipeDataExpert, iCurrentPWaferNum, bCassetteSuppleED);                            }                        }                        else                        {                            bWaferSuppleED = false;                            bEDShort = CassetteShortWaferInCassetteShort(pj, ref adjustedSlot, ref iCurrentEDWaferNum, layoutRecipeDataNormal, layoutRecipeDataExpert, bCassetteSuppleED, bWaferSuppleED);                        }                    }                }                else                {   // 只有在when P wafer short == leave as 的时候 when wafer in cassette are short 才执行判断                    if (strWhenPWaferShort == "LeaveAs")                    {                        if (strWhenWaferInCassetteAreShort == "LeaveAs")                        {                        }                        else if (strWhenWaferInCassetteAreShort == "SupplyED")                        {                            bCassetteSuppleED = false;                            bWaferSuppleED = true;                            bEDShort = CassetteShortWaferInCassetteShort(pj, ref adjustedSlot, ref iCurrentEDWaferNum, layoutRecipeDataNormal, layoutRecipeDataExpert, bCassetteSuppleED, bWaferSuppleED);                        }                        else                        {                            reason = "wafer in cassette are short";                            return true;                        }                    }                    else                    {                        bWaferSuppleED = false;                        bEDShort = CassetteShortPWaferShort(pj, ref adjustedSlot, ref iCurrentEDWaferNum, layoutRecipeDataNormal, layoutRecipeDataExpert, iCurrentPWaferNum, bWaferSuppleED);                    }                }            }            if (bSDMove)            {                for (int i = iDummyUpper; i < iSlotCount - iDummyLower; i++)                {                    if (adjustedSlot[i] != "")                    {                        for (int j = 0; j < iDummyUpper; j++)                        {                            adjustedSlot[i - j - 1] = "dummy";                        }                        break;                    }                }                for (int i = iSlotCount - iDummyLower - 1; i > iDummyUpper; i--)                {                    if (adjustedSlot[i] != "")                    {                        for (int j = 0; j < iDummyLower; j++)                        {                            adjustedSlot[i + j + 1] = "dummy";                        }                        break;                    }                }            }            else            {                if (bBoatSupplyED)                {                    for (int i = iDummyUpper; i < iSlotCount - iDummyLower; i++)                    {                        if (iCurrentEDWaferNum > 0)                        {                            if (adjustedSlot[i] == "")                            {                                adjustedSlot[i] = "extradummy";                                iCurrentEDWaferNum--;                            }                        }                        else                        {                            bEDShort = true;                            break;                        }                    }                }                for (int i = 0; i < iDummyUpper; i++)                {                    adjustedSlot[i] = "dummy";                }                for (int i = iSlotCount - iDummyLower; i < iSlotCount; i++)                {                    adjustedSlot[i] = "dummy";                }            }            if (bEDShort)            {                switch (strWhenEDAreShort)                {                    case "WarningAndContinue":                    case "AbortWhenError":                        reason = "extradummy wafer are short";                        break;                    case "LeaveAs":                        bEDShort = false;                        break;                }            }            sideDummySlots.Clear();            extraDummySlots.Clear();            monitor1Slots.Clear();            monitor2Slots.Clear();            productSlots.Clear();            for (int i = 0; i < adjustedSlot.Count; i++)            {                if (adjustedSlot[i].ToLower() == "dummy" && iCurrentSDWaferNum > 0)                {                    sideDummySlots.Add(i);                    iCurrentSDWaferNum--;                }                if (adjustedSlot[i].ToLower() == "extradummy")                {                    extraDummySlots.Add(i);                }                if (adjustedSlot[i].ToLower() == "monitor1" && iCurrentM1WaferNum > 0)                {                    monitor1Slots.Add(i);                    iCurrentM1WaferNum--;                }                if (adjustedSlot[i].ToLower() == "monitor2" && iCurrentM2WaferNum > 0)                {                    monitor2Slots.Add(i);                    iCurrentM2WaferNum--;                }                if (adjustedSlot[i].ToLower() == "pd")                {                    productSlots.Add(i);                }            }            bM1Short = (GetCurrentWaferNum(pj, WaferType.M1, out m1SlotSelect) < monitor1Slots.Count) || monitor1Slots.Count == 0;            bM2Short = (GetCurrentWaferNum(pj, WaferType.M2, out m2SlotSelect) < monitor2Slots.Count) || monitor2Slots.Count == 0;            bool bSDShort = (GetCurrentWaferNum(pj, WaferType.SD, out bool sdSlotSelect) < sideDummySlots.Count) || sideDummySlots.Count == 0;            if (SC.GetValue<bool>("System.IsLeaveAsAllWaferWhenWaferShort"))                bEDShort = bM1Short = bM2Short = bSDShort = false;            waferShort = bEDShort || bM1Short || bM2Short || bSDShort;            if (bM1Short || bM2Short)            {                if (reason == string.Empty)                {                    if (bSDShort)                    {                        reason = "dummy wafer and monitor wafer are short";                    }                    else                    {                        reason = "monitor wafer are short";                    }                }                else                {                    if (bSDShort)                    {                        reason = "extradummy wafer,monitor wafer and dummy wafer are short";                    }                    else                    {                        reason = "extradummy wafer and dummy wafer are short";                    }                }            }            else            {                if (reason == string.Empty)                {                    if (bSDShort)                    {                        reason = "dummy wafer are short";                    }                }                else                {                    if (bSDShort)                    {                        reason = "extradummy wafer and dummy wafer are short";                    }                }            }            if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")            {                sideDummySlots.Sort((x, y) => -x.CompareTo(y));//降序                extraDummySlots.Sort((x, y) => -x.CompareTo(y));//降序                monitor1Slots.Sort((x, y) => -x.CompareTo(y));//降序                monitor2Slots.Sort((x, y) => -x.CompareTo(y));//降序                productSlots.Sort((x, y) => -x.CompareTo(y));//降序            }            else            {                sideDummySlots.Sort((x, y) => x.CompareTo(y));//升序                extraDummySlots.Sort((x, y) => x.CompareTo(y));//升序                monitor1Slots.Sort((x, y) => x.CompareTo(y));//升序                monitor2Slots.Sort((x, y) => x.CompareTo(y));//升序                productSlots.Sort((x, y) => x.CompareTo(y));//升序            }            pj.SideDummySlots = sideDummySlots;            pj.ExtraDummySlots = extraDummySlots;            pj.Monitor1Slots = monitor1Slots;            pj.Monitor2Slots = monitor2Slots;            pj.ProductSlots = productSlots;            return true;        }        private bool CheckLayoutParameter(RecipeLayoutEntityNormal layoutRecipeDataNormal, out string reason)        {            reason = string.Empty;            if (string.IsNullOrEmpty(layoutRecipeDataNormal.DummyUpperSlot))            {                reason = "DummyUpper is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.DummyLowerSlot))            {                reason = "DummyLower is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.CenteringSlotPosition))            {                reason = "CenteringSlotPosition is null";                return false;            }            //if (string.IsNullOrEmpty(layoutRecipeDataNormal.Pitch))            //{            //    reason = "Pitch is null";            //    return false;            //}            if (string.IsNullOrEmpty(layoutRecipeDataNormal.ProductPosition))            {                reason = "ProductPosition is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.MonitorPosition))            {                reason = "MonitorPosition is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.WhenCassetteInBatchAreShort))            {                reason = "WhenCassetteInBatchAreShort is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.WhenPWaferShort))            {                reason = "WhenPWaferShort is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.WhenWaferInCassetteAreShort))            {                reason = "WhenWaferInCassetteAreShort is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.WhenEDAreShort))            {                reason = "WhenEDAreShort is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.SDRule))            {                reason = "SDRule is null";                return false;            }            if (string.IsNullOrEmpty(layoutRecipeDataNormal.RuleOfSpaceInBoat))            {                reason = "RuleOfSpaceInBoat is null";                return false;            }            return true;        }        private int GetCurrentCassetteNum(ProcessJobInfo pj, WaferType targetWaferType)        {            int iCassetteNum = 0;            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), targetWaferType.ToString()));            foreach (var target in _lstTotlaStockers)            {                if (pj == null || !pj.Stockers.Any(x => x.Item1 == target.Module.ToString()))                    continue;                var carrier = CarrierManager.Instance.GetCarrier(target.Module);                if (!carrier.IsEmpty && carrier.CarrierType == targetCarrierType)                    iCassetteNum++;            }            return iCassetteNum;        }        private int GetRemainWaferNum(ProcessJobInfo pj, WaferType targetWaferType)        {            int iWaferNum = 0;            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), targetWaferType.ToString()));            foreach (var target in _lstTotlaStockers)            {                var carrier = CarrierManager.Instance.GetCarrier(target.Module);                if (carrier == null || carrier.IsEmpty || carrier.CarrierType != targetCarrierType)                    continue;                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                if (pj == null || !pj.Stockers.Any(x => x.Item1 == target.Module.ToString()) || stockerInfo == null)                    continue;                var wafers = WaferManager.Instance.GetWafers(target.Module);                foreach (var wafer in wafers)                {                    if (!wafer.IsEmpty && wafer.WaferType == targetWaferType && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                        iWaferNum++;                }            }            if (CarrierManager.Instance.CheckHasCarrier(_carrierRobot.Module, 0))            {                var carrier = CarrierManager.Instance.GetCarrier(_carrierRobot.Module);                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                if (carrier.CarrierType == targetCarrierType && pj != null && pj.Stockers.Any(x => x.Item1 == carrier.InternalModuleName.ToString()) && stockerInfo != null)                {                    var wafers = WaferManager.Instance.GetWafers(_carrierRobot.Module);                    foreach (var wafer in wafers)                    {                        if (!wafer.IsEmpty && wafer.WaferType == targetWaferType && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                            iWaferNum++;                    }                }            }            foreach (var fims in _lstFIMSs)            {                if (!fims.IsOnline)                    continue;                var carrier = CarrierManager.Instance.GetCarrier(fims.Module);                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == carrier.InternalModuleName.ToString());                if (carrier.CarrierType == targetCarrierType && pj != null && pj.Stockers.Any(x => x.Item1 == carrier.InternalModuleName.ToString()) && stockerInfo != null)                {                    var wafers = WaferManager.Instance.GetWafers(fims.Module);                    foreach (var wafer in wafers)                    {                        if (!wafer.IsEmpty && wafer.WaferType == targetWaferType && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                            iWaferNum++;                    }                }            }            {                var wafers = WaferManager.Instance.GetWafers(ModuleName.PM1);                foreach (var wafer in wafers)                {                    if (wafer == null || wafer.IsEmpty || wafer.WaferType != targetWaferType)                        continue;                    var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == ((ModuleName)wafer.OriginStation).ToString());                    if (stockerInfo != null && pj != null && pj.Stockers.Any(x => x.Item1 == ((ModuleName)wafer.OriginStation).ToString()) && (!stockerInfo.Item3.Any() || stockerInfo.Item3.Any(x => x == wafer.OriginSlot)))                        iWaferNum++;                }            }            var useCount = 0;            if (targetWaferType == WaferType.M1)                useCount = pj.Monitor1Slots.Count;            if (targetWaferType == WaferType.M2)                useCount = pj.Monitor2Slots.Count;            return iWaferNum - useCount;        }        private int GetCurrentWaferNum(ProcessJobInfo pj, WaferType targetWaferType, out bool isSlotSelect)        {            int iWaferNum = 0;            isSlotSelect = false;            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), targetWaferType.ToString()));            foreach (var target in _lstTotlaStockers)            {                var carrier = CarrierManager.Instance.GetCarrier(target.Module);                if (carrier == null || carrier.IsEmpty || carrier.CarrierType != targetCarrierType)                    continue;                if (pj == null || !pj.Stockers.Any(x => x.Item1 == target.Module.ToString()))                    continue;                var stockerInfo = pj.Stockers.FirstOrDefault(x => x.Item1 == target.Module.ToString());                var wafers = WaferManager.Instance.GetWafers(target.Module);                foreach (var wafer in wafers)                {                    if (!stockerInfo.Item3.Any() && !wafer.IsEmpty && wafer.WaferType == targetWaferType)                        iWaferNum++;                    if (stockerInfo.Item3.Any(x => x == wafer.OriginSlot) && !wafer.IsEmpty && wafer.WaferType == targetWaferType)                    {                        iWaferNum++;                        isSlotSelect = true;                    }                }            }            return iWaferNum;        }        private int GetCurrentDummyWaferNum(WaferType targetWaferType)        {            int iWaferNum = 0;            var targetCarrierType = (CarrierType)(Enum.Parse(typeof(CarrierType), targetWaferType.ToString()));            foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)            {                if (module == ModuleName.SMIFA || module == ModuleName.SMIFB || module == ModuleName.Boat)                    continue;                if (ModuleHelper.IsStocker(module))                {                    var carrier = CarrierManager.Instance.GetCarrier(module);                    if (carrier == null || carrier.IsEmpty || carrier.CarrierType != targetCarrierType)                        continue;                }                var wafers = WaferManager.Instance.GetWafers(module);                if (wafers == null)                    continue;                foreach (var wafer in wafers)                {                    if (!wafer.IsEmpty && wafer.WaferType == targetWaferType)                        iWaferNum++;                }            }            return iWaferNum;        }        private bool CassetteShortWaferInCassetteShort(ProcessJobInfo pj, ref List<string> adjustslot, ref int currentEDNum, RecipeLayoutEntityNormal layoutRecipeDataNormal, RecipeLayoutEntityExpert layoutRecipeDataExpert, bool cassetteSupplyED, bool waferSupplyED)        {            bool bEDShort = false;            bool monitorSlot = layoutRecipeDataNormal.MonitorPosition.ToLower() == "slot";            int slotCount = SC.GetValue<int>($"Boat.SlotCount");            int cassetteSlotCount = SC.GetValue<int>("System.CassetteSlotCount");            adjustslot = new List<string>();            for (int i = 0; i < slotCount; i++)            {                adjustslot.Add("");            }            if (monitorSlot)            {                for (int i = 0; i < slotCount; i++)                {                    if (layoutRecipeDataExpert.Items[i].ToLower() == "monitor")                        adjustslot[i] = "monitor";                }            }            List<SchedulerStocker> tempStockers = new List<SchedulerStocker>();            foreach (var p in pj.Stockers)            {                var module = GetModule(p.Item1) as SchedulerStocker;                var carrier = CarrierManager.Instance.GetCarrier(module.Module);                if (!carrier.IsEmpty && carrier.CarrierType.ToString() == "P")                    tempStockers.Add(module);            }            if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")                tempStockers.Reverse();            int.TryParse(layoutRecipeDataNormal.DummyUpperSlot, out int dummyUpper);            int.TryParse(layoutRecipeDataNormal.DummyLowerSlot, out int dummyLower);            int.TryParse(layoutRecipeDataNormal.CenteringSlotPosition, out int centeringSlotPos);            int centeringSlotPosLower;            int centeringSlotPosUpper;            if (tempStockers.Count % 2 == 0)            {                centeringSlotPosLower = centeringSlotPos + (cassetteSlotCount + 1) * (tempStockers.Count / 2);                centeringSlotPosUpper = centeringSlotPos - (cassetteSlotCount + 1) * (tempStockers.Count / 2);            }            else            {                centeringSlotPosLower = centeringSlotPos + (cassetteSlotCount / 2) + 1 + (cassetteSlotCount + 1) * (tempStockers.Count / 2);                centeringSlotPosUpper = centeringSlotPos - (cassetteSlotCount / 2) - 1 - (cassetteSlotCount + 1) * (tempStockers.Count / 2);            }            if (centeringSlotPosLower >= slotCount - dummyLower)            {                int iIndex = slotCount - dummyLower - 1;                if (!monitorSlot)                {                    adjustslot[iIndex] = "monitor";                    iIndex--;                }                foreach (var stocker in tempStockers)                {                    var wafers = WaferManager.Instance.GetWafers(stocker.Module);                    if (wafers == null)                        continue;                    foreach (var wafer in wafers)                    {                        if (!wafer.IsEmpty)                        {                            for (int i = iIndex; i >= dummyUpper; i--)                            {                                if (adjustslot[i] == "")                                {                                    adjustslot[i] = "production";                                    iIndex = i - 1;                                    break;                                }                            }                        }                        else                        {                            if (waferSupplyED)                            {                                if (currentEDNum > 0)                                {                                    for (int i = iIndex; i >= dummyUpper; i--)                                    {                                        if (adjustslot[i] == "")                                        {                                            adjustslot[i] = "extradummy";                                            currentEDNum--;                                            iIndex = i - 1;                                            break;                                        }                                    }                                }                                else                                {                                    bEDShort = true;                                }                            }                        }                    }                    if (!monitorSlot)                    {                        adjustslot[iIndex] = "monitor";                        iIndex--;                    }                }                if (cassetteSupplyED && currentEDNum > 0)                {                    for (int i = iIndex; i >= dummyUpper; i--)                    {                        if (currentEDNum > 0)                        {                            if (adjustslot[i] == "")                            {                                adjustslot[i] = "extradummy";                                currentEDNum--;                            }                        }                        else                        {                            bEDShort = true;                            break;                        }                    }                }            }            else if (centeringSlotPosUpper <= dummyUpper + 1)            {                int iIndex = dummyUpper;                if (!monitorSlot)                {                    adjustslot[iIndex] = "monitor";                    iIndex++;                }                foreach (var stocker in tempStockers)                {                    var wafers = WaferManager.Instance.GetWafers(stocker.Module);                    if (wafers == null)                        continue;                    foreach (var wafer in wafers)                    {                        if (!wafer.IsEmpty)                        {                            for (int i = iIndex; i < slotCount - dummyLower; i++)                            {                                if (adjustslot[i] == "")                                {                                    adjustslot[i] = "production";                                    iIndex = i + 1;                                    break;                                }                            }                        }                        else                        {                            if (waferSupplyED)                            {                                if (currentEDNum > 0)                                {                                    for (int i = iIndex; i < slotCount - dummyLower; i++)                                    {                                        if (adjustslot[i] == "")                                        {                                            adjustslot[i] = "extradummy";                                            currentEDNum--;                                            iIndex = i + 1;                                            break;                                        }                                    }                                }                                else                                {                                    bEDShort = true;                                }                            }                        }                    }                    if (!monitorSlot)                    {                        adjustslot[iIndex] = "monitor";                        iIndex++;                    }                }                if (cassetteSupplyED && currentEDNum > 0)                {                    for (int i = iIndex; i < slotCount - dummyLower; i++)                    {                        if (currentEDNum > 0)                        {                            if (adjustslot[i] == "")                            {                                adjustslot[i] = "extradummy";                                currentEDNum--;                            }                        }                        else                        {                            bEDShort = true;                            break;                        }                    }                }            }            else            {                int iCount = 0;                int iUpperIndex = centeringSlotPos - 1;                int iLowerIndex = centeringSlotPos - 1;                int iUpperCount = 0;                int iLowerCount = 0;                bool bCassetteEvenNumber = tempStockers.Count % 2 == 0;                if (bCassetteEvenNumber && !monitorSlot)                {                    adjustslot[centeringSlotPos - 1] = "monitor";                    iUpperIndex--;                    iLowerIndex++;                    iCount++;                }                foreach (var stocker in tempStockers)                {                    var wafers = WaferManager.Instance.GetWafers(stocker.Module);                    if (wafers == null)                        continue;                    foreach (var wafer in wafers)                    {                        if (bCassetteEvenNumber)                        {                            if (!monitorSlot && iUpperCount >= cassetteSlotCount && iUpperIndex >= dummyUpper)                            {                                adjustslot[iUpperIndex] = "monitor";                                iUpperIndex--;                                iUpperCount = 0;                            }                            if (!monitorSlot && iLowerCount >= cassetteSlotCount && iLowerIndex < slotCount - dummyLower)                            {                                adjustslot[iLowerIndex] = "monitor";                                iLowerIndex++;                                iLowerCount = 0;                            }                        }                        else                        {                            if (!monitorSlot && iUpperCount + iLowerCount >= cassetteSlotCount && iUpperIndex >= dummyUpper && iLowerIndex < slotCount - dummyLower)                            {                                adjustslot[iUpperIndex] = "monitor";                                iUpperIndex--;                                iUpperCount = 0;                                adjustslot[iLowerIndex] = "monitor";                                iLowerIndex++;                                iLowerCount = 0;                                bCassetteEvenNumber = true;                            }                        }                        if (iCount % 2 == 0)                        {                            if (!wafer.IsEmpty)                            {                                for (int i = iUpperIndex; i >= dummyUpper; i--)                                {                                    if (adjustslot[i] == "")                                    {                                        adjustslot[i] = "production";                                        iUpperIndex = i - 1;                                        iUpperCount++;                                        iCount++;                                        break;                                    }                                }                            }                            else                            {                                if (waferSupplyED)                                {                                    if (currentEDNum > 0)                                    {                                        for (int i = iUpperIndex; i >= dummyUpper; i--)                                        {                                            if (adjustslot[i] == "")                                            {                                                adjustslot[i] = "extradummy";                                                currentEDNum--;                                                iUpperIndex = i - 1;                                                iUpperCount++;                                                iCount++;                                                break;                                            }                                        }                                    }                                    else                                    {                                        bEDShort = true;                                    }                                }                            }                        }                        else                        {                            if (!wafer.IsEmpty)                            {                                for (int i = iLowerIndex; i < slotCount - dummyLower; i++)                                {                                    if (adjustslot[i] == "")                                    {                                        adjustslot[i] = "production";                                        iLowerIndex = i + 1;                                        iLowerCount++;                                        iCount++;                                        break;                                    }                                }                            }                            else                            {                                if (waferSupplyED)                                {                                    if (currentEDNum > 0)                                    {                                        for (int i = iLowerIndex; i < slotCount - dummyLower; i++)                                        {                                            if (adjustslot[i] == "")                                            {                                                adjustslot[i] = "extradummy";                                                currentEDNum--;                                                iLowerIndex = i + 1;                                                iLowerCount++;                                                iCount++;                                                break;                                            }                                        }                                    }                                    else                                    {                                        bEDShort = true;                                    }                                }                            }                        }                    }                    if (!monitorSlot)                    {                        if (adjustslot[iUpperIndex + 1] != "M")                        {                            adjustslot[iUpperIndex] = "monitor";                            iUpperIndex--;                            iCount++;                        }                        if (adjustslot[iLowerIndex - 1] != "monitor")                        {                            adjustslot[iLowerIndex] = "monitor";                            iLowerIndex++;                            iCount++;                        }                    }                }                if (cassetteSupplyED && currentEDNum > 0)                {                    for (int i = 0; i < slotCount - iCount - dummyUpper - dummyLower; i++)                    {                        if (currentEDNum <= 0)                        {                            bEDShort = true;                            break;                        }                        for (int j = iUpperIndex; j >= dummyUpper; j--)                        {                            if (adjustslot[j] == "")                            {                                adjustslot[j] = "extradummy";                                iUpperIndex = j - 1;                                currentEDNum--;                                break;                            }                        }                        if (currentEDNum <= 0)                        {                            bEDShort = true;                            break;                        }                        for (int j = iLowerIndex; j < slotCount - dummyLower; j++)                        {                            if (adjustslot[j] == "")                            {                                adjustslot[j] = "extradummy";                                iLowerIndex = j + 1;                                currentEDNum--;                                break;                            }                        }                    }                }            }            return bEDShort;        }        private bool CassetteShortPWaferShort(ProcessJobInfo pj, ref List<string> adjustslot, ref int currentEDNum, RecipeLayoutEntityNormal layoutRecipeDataNormal, RecipeLayoutEntityExpert layoutRecipeDataExpert, int currentPWaferNum, bool cassetteSupplyED)        {            bool bEDShort = false;            bool monitorSlot = layoutRecipeDataNormal.MonitorPosition.ToLower() == "slot";            int slotCount = SC.GetValue<int>($"Boat.SlotCount");            int cassetteSlotCount = SC.GetValue<int>("System.CassetteSlotCount");            adjustslot = new List<string>();            for (int i = 0; i < slotCount; i++)            {                adjustslot.Add("");            }            if (monitorSlot)            {                for (int i = 0; i < slotCount; i++)                {                    if (layoutRecipeDataExpert.Items[i].ToLower() == "monitor")                        adjustslot[i] = "monitor";                }            }            int.TryParse(layoutRecipeDataNormal.DummyUpperSlot, out int dummyUpper);            int.TryParse(layoutRecipeDataNormal.DummyLowerSlot, out int dummyLower);            int.TryParse(layoutRecipeDataNormal.CenteringSlotPosition, out int centeringSlotPos);            List<SchedulerStocker> tempStockers = new List<SchedulerStocker>();            foreach (var p in pj.Stockers)            {                var module = GetModule(p.Item1) as SchedulerStocker;                var carrier = CarrierManager.Instance.GetCarrier(module.Module);                if (!carrier.IsEmpty && carrier.CarrierType.ToString() == "P")                    tempStockers.Add(module);            }            if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")                tempStockers.Reverse();            int centeringSlotPosLower;            int centeringSlotPosUpper;            if ((currentPWaferNum / 2) % cassetteSlotCount == 0)            {                var temp = (int)Math.Ceiling((float)((currentPWaferNum / 2) / cassetteSlotCount));                centeringSlotPosLower = centeringSlotPos + (cassetteSlotCount + 1) * temp;                centeringSlotPosUpper = centeringSlotPos - (cassetteSlotCount + 1) * temp;            }            else            {                var temp = (int)Math.Ceiling((float)((currentPWaferNum / 2) / cassetteSlotCount));                centeringSlotPosLower = centeringSlotPos + (currentPWaferNum / 2) + 1 + temp;                centeringSlotPosUpper = centeringSlotPos - (currentPWaferNum / 2) - 1 - temp;            }            if (centeringSlotPosLower >= slotCount - dummyLower)            {                int iIndex = slotCount - dummyLower - 1;                if (!monitorSlot)                {                    adjustslot[iIndex] = "monitor";                    iIndex--;                }                for (int i = 0; i < tempStockers.Count; i++)                {                    if (currentPWaferNum == 0)                        break;                    for (int j = 0; j < cassetteSlotCount; j++)                    {                        if (currentPWaferNum == 0)                            break;                        for (int k = iIndex; k >= dummyUpper; k--)                        {                            if (adjustslot[k] == "")                            {                                adjustslot[k] = "production";                                currentPWaferNum--;                                iIndex = k - 1;                                break;                            }                        }                    }                    if (!monitorSlot)                    {                        adjustslot[iIndex] = "monitor";                        iIndex--;                    }                }                if (cassetteSupplyED)                {                    for (int i = iIndex; i >= dummyUpper; i--)                    {                        if (currentEDNum > 0)                        {                            if (adjustslot[i] == "")                            {                                adjustslot[i] = "extradummy";                                currentEDNum--;                            }                        }                        else                        {                            bEDShort = true;                            break;                        }                    }                }            }            else if (centeringSlotPosUpper <= dummyUpper + 1)            {                int iIndex = dummyUpper;                if (!monitorSlot)                {                    adjustslot[iIndex] = "monitor";                    iIndex++;                }                for (int i = 0; i < tempStockers.Count; i++)                {                    if (currentPWaferNum == 0)                        break;                    for (int j = 0; j < cassetteSlotCount; j++)                    {                        if (currentPWaferNum == 0)                            break;                        for (int k = iIndex; k < slotCount - dummyLower; k++)                        {                            if (adjustslot[k] == "")                            {                                adjustslot[k] = "production";                                currentPWaferNum--;                                iIndex = k + 1;                                break;                            }                        }                    }                    if (!monitorSlot)                    {                        adjustslot[iIndex] = "monitor";                        iIndex++;                    }                }                if (cassetteSupplyED)                {                    for (int i = iIndex; i < slotCount - dummyLower; i++)                    {                        if (currentEDNum > 0)                        {                            if (adjustslot[i] == "")                            {                                adjustslot[i] = "extradummy";                                currentEDNum--;                            }                        }                        else                        {                            bEDShort = true;                            break;                        }                    }                }            }            else            {                int iCount = 0;                int iUpperIndex = centeringSlotPos - 1;                int iLowerIndex = centeringSlotPos - 1;                int iUpperCount = 0;                int iLowerCount = 0;                bool bCassetteEvenNumber = (currentPWaferNum / 2) % cassetteSlotCount == 0;                if (bCassetteEvenNumber && !monitorSlot)                {                    adjustslot[centeringSlotPos - 1] = "monitor";                    iUpperIndex--;                    iLowerIndex++;                    iCount++;                }                for (int i = 0; i < tempStockers.Count; i++)                {                    if (currentPWaferNum == 0)                        break;                    for (int j = 0; j < cassetteSlotCount; j++)                    {                        if (currentPWaferNum == 0)                            break;                        if (bCassetteEvenNumber)                        {                            if (!monitorSlot && iUpperCount >= cassetteSlotCount && iUpperIndex >= dummyUpper)                            {                                adjustslot[iUpperIndex] = "monitor";                                iUpperIndex--;                                iUpperCount = 0;                            }                            if (!monitorSlot && iLowerCount >= cassetteSlotCount && iLowerIndex < slotCount - dummyLower)                            {                                adjustslot[iLowerIndex] = "monitor";                                iLowerIndex++;                                iLowerCount = 0;                            }                        }                        else                        {                            if (!monitorSlot && iUpperCount + iLowerCount >= cassetteSlotCount && iUpperIndex >= dummyUpper && iLowerIndex < slotCount - dummyLower)                            {                                adjustslot[iUpperIndex] = "monitor";                                iUpperIndex--;                                iUpperCount = 0;                                adjustslot[iLowerIndex] = "monitor";                                iLowerIndex++;                                iLowerCount = 0;                                bCassetteEvenNumber = true;                            }                        }                        if (iCount % 2 == 0)                        {                            for (int k = iUpperIndex; k >= dummyUpper; k--)                            {                                if (adjustslot[k] == "")                                {                                    adjustslot[k] = "production";                                    currentPWaferNum--;                                    iUpperIndex = k - 1;                                    iUpperCount++;                                    iCount++;                                    break;                                }                            }                        }                        else                        {                            for (int k = iLowerIndex; k < slotCount - dummyLower; k++)                            {                                if (adjustslot[k] == "")                                {                                    adjustslot[k] = "production";                                    currentPWaferNum--;                                    iLowerIndex = k + 1;                                    iLowerCount++;                                    iCount++;                                    break;                                }                            }                        }                    }                }                if (!monitorSlot)                {                    if (adjustslot[iUpperIndex + 1] != "monitor")                    {                        adjustslot[iUpperIndex] = "monitor";                        iUpperIndex--;                        iCount++;                    }                    if (adjustslot[iLowerIndex - 1] != "monitor")                    {                        adjustslot[iLowerIndex] = "monitor";                        iLowerIndex++;                        iCount++;                    }                }                if (cassetteSupplyED)                {                    for (int i = 0; i < slotCount - iCount - dummyUpper - dummyLower; i++)                    {                        if (currentEDNum <= 0)                        {                            bEDShort = true;                            break;                        }                        for (int j = iUpperIndex; j >= dummyUpper; j--)                        {                            if (adjustslot[j] == "")                            {                                adjustslot[j] = "extradummy";                                currentEDNum--;                                iUpperIndex = j - 1;                                break;                            }                        }                        if (currentEDNum <= 0)                        {                            bEDShort = true;                            break;                        }                        for (int j = iLowerIndex; j < slotCount - dummyLower; j++)                        {                            if (adjustslot[j] == "")                            {                                adjustslot[j] = "extradummy";                                currentEDNum--;                                iLowerIndex = j + 1;                                break;                            }                        }                    }                }            }            return bEDShort;        }        #endregion        #region dummy use         private bool CheckDummyWaferUse(ProcessJobInfo pj, ControlJobInfo cj)        {            var pm = (Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule);            foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)            {                if (!ModuleHelper.IsStocker(module) || !pj.Stockers.Any(x => x.Item1 == module.ToString()))                    continue;                var wafers = WaferManager.Instance.GetWafers(module);                if (wafers == null)                    continue;                foreach (var wafer in wafers)                {                    if (wafer.IsEmpty)                        continue;                    #region New                    var isScheduleMaintenance = SC.ContainsItem("System.IsScheduleMaintenance") && SC.GetValue<bool>("System.IsScheduleMaintenance");                    if (isScheduleMaintenance)                    {                        #region  SD Dummy 时间/次数/膜厚限制                        var maintenanceItemSDCarrierUseFreq = ScheduleMaintenanceDataManager.Instance.GetItem(MaintenanceItemEnum.SDCarrierUseFreq.ToString());                        var maintenanceItemSDWaferThickness = ScheduleMaintenanceDataManager.Instance.GetItem(MaintenanceItemEnum.SDWaferThickness.ToString());                        var maintenanceItemSDWaferUserTime = ScheduleMaintenanceDataManager.Instance.GetItem(MaintenanceItemEnum.SDWaferUserTime.ToString());                        int SDCountWarning = (int)maintenanceItemSDCarrierUseFreq.StartValue;                        int SDCountAlarm = (int)maintenanceItemSDCarrierUseFreq.LimitValue;                        bool IsEnableSDWaferUseCountWarning = SDCountWarning > 0;                        int SDThicknessWarning = (int)maintenanceItemSDWaferThickness.StartValue;                        int SDThicknessAlarm = (int)maintenanceItemSDWaferThickness.LimitValue;                        bool IsEnableSDWaferThicknessWarning = SDThicknessWarning > 0;                        int SDTimeWarning = DateTimeUtil.GetSecond(maintenanceItemSDWaferUserTime.StartValue);                        int SDTimeAlarm = DateTimeUtil.GetSecond(maintenanceItemSDWaferUserTime.LimitValue);                        bool IsEnableSDWaferUseTimeWarning = SDTimeWarning > 0;                        if (wafer.WaferType == WaferType.SD && IsEnableSDWaferUseCountWarning && wafer.UseCount > SDCountWarning && wafer.UseCount < SDCountAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemSDCarrierUseFreq.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.SDWaferTypeUseCountWarning.Set, $"more than {SDCountWarning} 次"));                        }                        if (wafer.WaferType == WaferType.SD && IsEnableSDWaferUseTimeWarning && wafer.UseTime > SDTimeWarning && wafer.UseTime < SDTimeAlarm)                        {                            DateTimeUtil.ConvertSecondsToHoursAndMinutes(SDTimeWarning, out var hours, out var minutes);                            pm.MaintenanceProcessingCommandExec(maintenanceItemSDWaferUserTime.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.SDWaferTypeUseTimeWarning.Set, $"more than {hours} hour {minutes} min"));                        }                        if (wafer.WaferType == WaferType.SD && IsEnableSDWaferThicknessWarning && wafer.UseThick > SDThicknessWarning && wafer.UseThick < SDThicknessAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemSDWaferThickness.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.SDWaferTypeThicknessWarning.Set, $"more than {SDThicknessWarning} nm"));                        }                        if (wafer.WaferType == WaferType.SD && IsEnableSDWaferUseCountWarning && wafer.UseCount >= SDCountAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemSDCarrierUseFreq.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.SDWaferTypeUseCountAlarm.Set, $"more than {SDCountAlarm} 次"));                            return false;                        }                        if (wafer.WaferType == WaferType.SD && IsEnableSDWaferUseTimeWarning && wafer.UseTime >= SDTimeAlarm)                        {                            DateTimeUtil.ConvertSecondsToHoursAndMinutes(SDTimeAlarm, out var hours, out var minutes);                            pm.MaintenanceProcessingCommandExec(maintenanceItemSDWaferUserTime.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.SDWaferTypeUseTimeAlarm.Set, $"more than {hours} hour {minutes}"));                            return false;                        }                        if (wafer.WaferType == WaferType.SD && IsEnableSDWaferThicknessWarning && wafer.UseThick >= SDThicknessAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemSDWaferThickness.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.SDWaferTypeThicknessAlarm.Set, $"more than {SDThicknessAlarm} nm"));                            return false;                        }                        #endregion                        #region ED                        var maintenanceItemEDCarrierUseFreq = ScheduleMaintenanceDataManager.Instance.GetItem(MaintenanceItemEnum.FDCarrierUseFreq.ToString());                        var maintenanceItemEDWaferThickness = ScheduleMaintenanceDataManager.Instance.GetItem(MaintenanceItemEnum.FDWaferThickness.ToString());                        var maintenanceItemEDWaferUserTime = ScheduleMaintenanceDataManager.Instance.GetItem(MaintenanceItemEnum.FDWaferUserTime.ToString());                        int EDCountWarning = (int)maintenanceItemEDCarrierUseFreq.StartValue;                        int EDCountAlarm = (int)maintenanceItemEDCarrierUseFreq.LimitValue;                        bool IsEnableEDWaferUseCountWarning = EDCountWarning > 0;                        int EDThicknessWarning = (int)maintenanceItemEDWaferThickness.StartValue;                        int EDThicknessAlarm = (int)maintenanceItemEDWaferThickness.LimitValue;                        bool IsEnableEDWaferThicknessWarning = EDThicknessWarning > 0;                        int EDTimeWarning = DateTimeUtil.GetSecond(maintenanceItemEDWaferUserTime.StartValue);                        int EDTimeAlarm = DateTimeUtil.GetSecond(maintenanceItemEDWaferUserTime.LimitValue);                        bool IsEnableEDWaferUseTimeWarning = EDTimeWarning > 0;                        if (wafer.WaferType == WaferType.ED && IsEnableEDWaferUseCountWarning && maintenanceItemEDCarrierUseFreq.CurrentValue > EDCountWarning && wafer.UseCount < EDCountAlarm)                        {                            //Singleton<ScheduleMaintenanceDataManager>.Instance.MaintenanceProcessingCommandExec(maintenanceItemEDCarrierUseFreq.MaintenanceProcessing, Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountWarning.Set, $"more than {EDCountWarning} 次");                            //return false;                            pm.MaintenanceProcessingCommandExec(                                maintenanceItemEDCarrierUseFreq.MaintenanceProcessing,                                new AlarmReportItem(Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountWarning.Set, $"more than {EDCountWarning} 次")                                );                            //if (Enum.TryParse<MaintenanceProcessingCommandEnum>(maintenanceItemEDCarrierUseFreq.MaintenanceProcessing, out var command))                            //{                            //    switch (command)                            //    {                            //        case MaintenanceProcessingCommandEnum.None:                            //            break;                            //        case MaintenanceProcessingCommandEnum.AlarmReport:                            //            Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountWarning.Set($"more than {EDCountWarning} 次");                            //            break;                            //        case MaintenanceProcessingCommandEnum.JobProhibition:                            //            break;                            //        case MaintenanceProcessingCommandEnum.JobManualStart:                            //            break;                            //        case MaintenanceProcessingCommandEnum.JobAutoStart:                            //            Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountWarning.Set($"more than {EDCountWarning} 次");                            //            Singleton<EquipmentManager>.Instance.CheckToPostMessage((int)EquipmentManager.MSG.JobAutoStart, cj.Name);                            //            var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                            //            if (pmModule != null)                            //                pmModule.Process(maintenanceItemEDCarrierUseFreq.AssociationProcessRecipeName, false, true, out string reson);                            //            return false;                            //    }                            //}                        }                        if (wafer.WaferType == WaferType.ED && IsEnableEDWaferUseTimeWarning && wafer.UseTime > EDTimeWarning && wafer.UseTime < EDTimeAlarm)                        {                            DateTimeUtil.ConvertSecondsToHoursAndMinutes(EDTimeWarning, out var hours, out var minutes);                            pm.MaintenanceProcessingCommandExec(maintenanceItemEDWaferUserTime.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.EDWaferTypeUseTimeWarning.Set, $"more than {hours} hour {minutes} min"));                        }                        if (wafer.WaferType == WaferType.ED && IsEnableEDWaferThicknessWarning && wafer.UseThick > EDThicknessWarning && wafer.UseThick < EDThicknessAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemEDWaferThickness.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.EDWaferTypeThicknessWarning.Set, $"more than {EDThicknessWarning} nm"));                        }                        if (wafer.WaferType == WaferType.ED && IsEnableEDWaferUseCountWarning && wafer.UseCount >= EDCountAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemEDCarrierUseFreq.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountAlarm.Set, $"more than {EDCountAlarm} 次"));                            return false;                        }                        if (wafer.WaferType == WaferType.ED && IsEnableEDWaferUseTimeWarning && wafer.UseTime >= EDTimeAlarm)                        {                            DateTimeUtil.ConvertSecondsToHoursAndMinutes(EDTimeAlarm, out var hours, out var minutes);                            pm.MaintenanceProcessingCommandExec(maintenanceItemEDWaferUserTime.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.EDWaferTypeUseTimeAlarm.Set, $"more than {hours} hour {minutes}"));                            return false;                        }                        if (wafer.WaferType == WaferType.ED && IsEnableEDWaferThicknessWarning && wafer.UseThick >= EDThicknessAlarm)                        {                            pm.MaintenanceProcessingCommandExec(maintenanceItemEDWaferThickness.MaintenanceProcessing, new AlarmReportItem(Singleton<EquipmentManager>.Instance.EDWaferTypeThicknessAlarm.Set, $"more than {EDThicknessAlarm} nm"));                            return false;                        }                        #endregion                    }                    else                    {                        // SD Dummy                        if (wafer.WaferType == WaferType.SD && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableSDWaferUseCountWarning") && wafer.UseCount > SC.GetValue<int>($"PM1.WaferCycleTime.SDCountWarning") && wafer.UseCount < SC.GetValue<int>($"PM1.WaferCycleTime.SDCountAlarm"))                        {                            Singleton<EquipmentManager>.Instance.SDWaferTypeUseCountWarning.Set($"more than {SC.GetValue<int>($"PM1.WaferCycleTime.SDCountWarning")} 次");                        }                        if (wafer.WaferType == WaferType.SD && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableSDWaferUseTimeWarning") && wafer.UseTime / 3600 > SC.GetValue<double>($"PM1.WaferCycleTime.SDTimeWarning") && wafer.UseTime / 3600 < SC.GetValue<double>($"PM1.WaferCycleTime.SDTimeAlarm"))                        {                            Singleton<EquipmentManager>.Instance.SDWaferTypeUseTimeWarning.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.SDTimeWarning")} hour");                        }                        if (wafer.WaferType == WaferType.SD && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableSDWaferThicknessWarning") && wafer.UseThick > SC.GetValue<double>($"PM1.WaferCycleTime.SDThicknessWarning") && wafer.UseThick < SC.GetValue<double>($"PM1.WaferCycleTime.SDThicknessAlarm"))                        {                            Singleton<EquipmentManager>.Instance.SDWaferTypeThicknessWarning.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.SDThicknessWarning")} nm");                        }                        if (wafer.WaferType == WaferType.SD && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableSDWaferUseCountWarning") && wafer.UseCount >= SC.GetValue<int>($"PM1.WaferCycleTime.SDCountAlarm"))                        {                            Singleton<EquipmentManager>.Instance.SDWaferTypeUseCountAlarm.Set($"more than {SC.GetValue<int>($"PM1.WaferCycleTime.SDCountAlarm")} 次");                            return false;                        }                        if (wafer.WaferType == WaferType.SD && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableSDWaferUseTimeWarning") && wafer.UseTime / 3600 >= SC.GetValue<double>($"PM1.WaferCycleTime.SDTimeAlarm"))                        {                            Singleton<EquipmentManager>.Instance.SDWaferTypeUseTimeAlarm.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.SDTimeAlarm")} hour");                            return false;                        }                        if (wafer.WaferType == WaferType.SD && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableSDWaferThicknessWarning") && wafer.UseThick >= SC.GetValue<double>($"PM1.WaferCycleTime.SDThicknessAlarm"))                        {                            Singleton<EquipmentManager>.Instance.SDWaferTypeThicknessAlarm.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.SDThicknessAlarm")} nm");                            return false;                        }                        // Extra Dummy                        if (wafer.WaferType == WaferType.ED && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableEDWaferUseCountWarning") && wafer.UseCount > SC.GetValue<int>($"PM1.WaferCycleTime.EDCountWarning") && wafer.UseCount < SC.GetValue<int>($"PM1.WaferCycleTime.EDCountAlarm"))                        {                            Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountWarning.Set($"more than {SC.GetValue<int>($"PM1.WaferCycleTime.EDCountWarning")} 次");                        }                        if (wafer.WaferType == WaferType.ED && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableEDWaferUseTimeWarning") && wafer.UseTime / 3600 > SC.GetValue<double>($"PM1.WaferCycleTime.EDTimeWarning") && wafer.UseTime / 3600 < SC.GetValue<double>($"PM1.WaferCycleTime.EDTimeAlarm"))                        {                            Singleton<EquipmentManager>.Instance.EDWaferTypeUseTimeWarning.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.EDTimeWarning")} hour");                        }                        if (wafer.WaferType == WaferType.ED && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableEDWaferThicknessWarning") && wafer.UseThick > SC.GetValue<double>($"PM1.WaferCycleTime.EDThicknessWarning") && wafer.UseThick < SC.GetValue<double>($"PM1.WaferCycleTime.EDThicknessAlarm"))                        {                            Singleton<EquipmentManager>.Instance.EDWaferTypeThicknessWarning.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.EDThicknessWarning")} nm");                        }                        if (wafer.WaferType == WaferType.ED && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableEDWaferUseCountWarning") && wafer.UseCount >= SC.GetValue<int>($"PM1.WaferCycleTime.EDCountAlarm"))                        {                            Singleton<EquipmentManager>.Instance.EDWaferTypeUseCountAlarm.Set($"more than {SC.GetValue<int>($"PM1.WaferCycleTime.EDCountAlarm")} 次");                            return false;                        }                        if (wafer.WaferType == WaferType.ED && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableEDWaferUseTimeWarning") && wafer.UseTime / 3600 >= SC.GetValue<double>($"PM1.WaferCycleTime.EDTimeAlarm"))                        {                            Singleton<EquipmentManager>.Instance.EDWaferTypeUseTimeAlarm.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.EDTimeAlarm")} hour");                            return false;                        }                        if (wafer.WaferType == WaferType.ED && SC.GetValue<bool>($"PM1.WaferCycleTime.IsEnableEDWaferThicknessWarning") && wafer.UseThick >= SC.GetValue<double>($"PM1.WaferCycleTime.EDThicknessAlarm"))                        {                            Singleton<EquipmentManager>.Instance.EDWaferTypeThicknessAlarm.Set($"more than {SC.GetValue<double>($"PM1.WaferCycleTime.EDThicknessAlarm")} nm");                            return false;                        }                    }                    #endregion                }            }            return true;        }        #endregion        private void MonitorPMTask()        {            if (!_carrierRobot.IsAvailable || !_waferRobot.IsAvailable)                return;            foreach (var pm in _lstPms)            {                var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;                if (pmModule == null)                    return;                if (!pm.IsAvailable && !pmModule.IsMainRecipeComplete)                    continue;                var wafers = WaferManager.Instance.GetWafers(pm.Module);                if (wafers == null)                    continue;                var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);                if (pj == null)                    return;                bool isNeedProcess = false;                bool isProcessed = false;                foreach (var wafer in wafers)                {                    if (wafer == null || wafer.IsEmpty)                        continue;                    if (!pj.SlotWafers.Any(x => x.Item1 == (ModuleName)wafer.OriginStation && x.Item2 == wafer.OriginSlot))                        pj.SlotWafers.Add(Tuple.Create((ModuleName)wafer.OriginStation, wafer.OriginSlot));                    if (wafer.ProcessJob == null)                    {                        wafer.ProcessJob = pj;                        WaferDataRecorder.SetPjInfo(wafer.InnerId.ToString(), pj.InnerId.ToString());                    }                    if (wafer.ProcessState == EnumWaferProcessStatus.Idle)                    {                        isNeedProcess = true;                        isProcessed = false;                    }                    if (wafer.ProcessState == EnumWaferProcessStatus.Completed)                    {                        isNeedProcess = false;                        isProcessed = true;                    }                }                if (pj.IsEmptyLayout)                {                    if (pj.ProcessingState == EnumProcessingState.Charging)                    {                        isNeedProcess = true;                        isProcessed = false;                    }                    else                    {                        isNeedProcess = false;                        if (DateTime.Now - pm.RecipeStartTime > TimeSpan.FromSeconds(5))                            isProcessed = true;                    }                }                if (isNeedProcess && CheckChargeFinished(pj, pm.Module) && pj.ProcessingState != EnumProcessingState.Processing &&                    pmModule.CheckStandbyCondition() && pm.IsAvailable)                {                    if (SC.GetValue<bool>("System.IsEnableBoatChargeMapping") && !pj.IsMapAfterCharge)                    {                        var mappingSlot = GetMappingSlot(ModuleName.PM1);                        if (_waferRobot.Map(ModuleName.PM1, mappingSlot.Length == 0 ? 0 : Regex.Matches(mappingSlot, "1").Count, mappingSlot, true, false, Hand.Blade1))                        {                            pj.IsMapAfterCharge = true;                            return;                        }                    }                    if (!pj.RobotHomed)                    {                        (Singleton<EquipmentManager>.Instance.Modules[ModuleName.WaferRobot] as WaferRobotModule).Home(out _);                        pj.RobotHomed = true;                    }                    pj.ProcessingState = EnumProcessingState.Processing;                    pmModule?.SetN2PurgeLAO2CheckFirstEnable(false);                    if (pm.Process((string)pj.ProcessRecipe, false, true))                        return;                }                //工艺结束                if (isProcessed)                {                    if (pj.ProcessingState == EnumProcessingState.Processing)                    {                        Singleton<EquipmentManager>.Instance.EndPj();                        if (pj.CoolTimeSec > 0)                        {                            _coolTimer.Restart();                            pj.ProcessingState = EnumProcessingState.CoolingBeforeDischarge;                        }                        else                        {                            _coolTimer.Restart();                            pj.ProcessingState = EnumProcessingState.CoolingBeforeDischarge;                        }                        LOG.Write($"Cooling start {pj.CoolTimeSec}");                    }                    if (pj.ProcessingState == EnumProcessingState.CoolingBeforeDischarge && (_coolTimer.ElapsedMilliseconds > pj.CoolTimeSec * 1000 || _isCoolingSkip))                    {                        _isCoolingSkip = false;                        var boatModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.Boat] as BoatModule;                        if (boatModule != null && !boatModule.IsBoatElevatorAtHomePosition && boatModule.ZAxisDevice.IsReady)                        {                            boatModule.ZAxisDevice.SetServoMoveTo("HomePosition", out _);                        }                        if (boatModule != null && !boatModule.IsBoatRotationAtHomePosition && boatModule.RAxisDevice.IsReady)                        {                            boatModule.RAxisDevice.ServoStop();                            boatModule.RAxisDevice.SetServoHome();                        }                        if (boatModule != null && boatModule.IsBoatElevatorAtHomePosition && boatModule.IsBoatRotationAtHomePosition)                        {                            if (SC.GetValue<bool>("System.IsEnableBoatDischargeMapping") && !pj.IsMapBeforeDischarge)                            {                                var mappingSlot = GetMappingSlot(ModuleName.PM1);                                if (_waferRobot.Map(ModuleName.PM1, mappingSlot.Length == 0 ? 0 : Regex.Matches(mappingSlot, "1").Count, mappingSlot, true, false, Hand.Blade1))                                {                                    pj.IsMapBeforeDischarge = true;                                    return;                                }                            }                            _coolTimer.Stop();                            AdjustDischargeSlots();                            pj.ProcessingState = EnumProcessingState.Discharging;                            LOG.Write($"Cooling completed {pj.CoolTimeSec}");                            if (pj.IsN2PurgeMode)                            {                                //pmModule?.SetN2PurgeProcess(true);                                //if (SC.ContainsItem("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable") && SC.GetValue<bool>("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable"))                                //    pmModule?.SetN2PurgeLAO2CheckFirstEnable(true);                                if (SC.ContainsItem("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable") && SC.GetValue<bool>("PM1.N2Purge.WaferCharge.TransferRoomN2PurgeEnable"))                                {                                    pmModule?.SetN2PurgeProcess(true);                                    pmModule?.SetN2PurgeLAO2CheckFirstEnable(true);                                }                                if (SC.ContainsItem("PM1.RecipeRelevancyN2Purge") && SC.GetValue<bool>("PM1.RecipeRelevancyN2Purge") && SC.ContainsItem("PM1.SelectN2PurgeMode"))                                {                                    var setValue = pmModule.GetN2PurgeModeEnumByStr(pj.N2PurgeModeStr);                                    SC.SetItemValueFromString("PM1.SelectN2PurgeMode", setValue);                                    pmModule.RestAllN2PrugeRD();                                }                            }                        }                    }                }            }        }        #endregion Module task        #region Logic Check        private bool CheckChargeFinished(ProcessJobInfo pj, ModuleName module, bool isOnlyCharge = false)        {            var wafers = WaferManager.Instance.GetWafers(module);            if (pj == null || wafers == null)                return false;            if (WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 0) ||                WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 1) ||                WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 2) ||                WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 3) ||                 WaferManager.Instance.CheckHasWafer(ModuleName.WaferRobot, 4))                return false;            int edCount = 0;            int sdCount = 0;            int m1Count = 0;            int m2Count = 0;            int pCount = 0;            int expertCount = 0;            foreach (var wafer in wafers)            {                if (wafer == null || wafer.IsEmpty)                    continue;                switch (wafer.WaferType)                {                    case WaferType.M1:                        m1Count++;                        break;                    case WaferType.M2:                        m2Count++;                        break;                    case WaferType.ED:                        edCount++;                        break;                    case WaferType.SD:                        sdCount++;                        break;                    case WaferType.P:                        pCount++;                        break;                    case WaferType.XD:                    case WaferType.T:                        expertCount++;                        break;                }            }            if (expertCount > 0)            {                if (pj.XdSlots == null || pj.TSlots == null)                    return false;                return expertCount == pj.XdSlots.Count + pj.TSlots.Count;            }            if (pj.ExtraDummySlots != null && pj.ExtraDummySlots.Count != edCount && pj.ExtraDummySlots.Count > 0)                return false;            if (pj.SideDummySlots != null && pj.SideDummySlots.Count != sdCount && pj.SideDummySlots.Count > 0)                return false;            if (pj.Monitor1Slots != null && pj.Monitor1Slots.Count != m1Count)                return false;            if (pj.Monitor2Slots != null && pj.Monitor2Slots.Count != m2Count)                return false;            foreach (var s in pj.Stockers)            {                var carrier = CarrierManager.Instance.GetCarrier(s.Item1, 0);                if (carrier == null || carrier.CarrierType != CarrierType.P || carrier.IsEmpty)                    continue;                var carrierWafers = WaferManager.Instance.GetWafers(ModuleHelper.Converter(s.Item1)).ToList();                if (carrierWafers == null || (carrierWafers.Any(x => !x.IsEmpty) && pj.ProductSlots.Count > pCount))                    return false;            }            var carrierRobot = CarrierManager.Instance.GetCarrier(_carrierRobot.Module, 0);            if (carrierRobot != null && !carrierRobot.IsEmpty && isOnlyCharge)                return false;            if (carrierRobot != null && !carrierRobot.IsEmpty &&                carrierRobot.CarrierType == CarrierType.P &&                pj.Stockers.Any(x => ModuleHelper.Converter(x.Item1) == carrierRobot.InternalModuleName))            {                var carrierWafers = WaferManager.Instance.GetWafers(_carrierRobot.Module).ToList();                if (carrierWafers == null || carrierWafers.Any(x => !x.IsEmpty))                    return false;            }            foreach (var fims in _lstFIMSs)            {                var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                if (carrier != null && !carrier.IsEmpty && isOnlyCharge)                    return false;                if (carrier != null && !carrier.IsEmpty &&                    carrier.CarrierType == CarrierType.P &&                    pj.Stockers.Any(x => ModuleHelper.Converter(x.Item1) == carrier.InternalModuleName))                {                    var carrierWafers = WaferManager.Instance.GetWafers(fims.Module).ToList();                    if (carrierWafers == null || carrierWafers.Any(x => !x.IsEmpty))                        return false;                }            }            return true;        }        private bool CheckProcessJobReady(ProcessJobInfo pj)        {            bool isReady = true;            if (pj == null)                return false;            //Boat Ready检查            var pmModule = Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule;            if (pmModule == null || !pmModule.CheckReadyForTransfer(_waferRobot.Module, Hand.Blade1, 0, EnumTransferType.Place, out string reason))                return false;            int pCount = 0;            int m1Count = 0;            int m2Count = 0;            foreach (var s in pj.Stockers)            {                var type = SC.GetStringValue($"System.Stocker.{s.Item1}WaferType");                if (!string.IsNullOrEmpty(type) && (type.Contains("P") || type.Contains("M") || type.Contains("None")))                {                    ModuleName module = ModuleHelper.Converter(s.Item1);                    if (!CarrierManager.Instance.CheckHasCarrier(module, 0))                        continue;                    var wafers = WaferManager.Instance.GetWafers(module).ToList();                    if (wafers == null)                        continue;                    foreach (var wafer in wafers)                    {                        if (wafer == null || wafer.ProcessState != EnumWaferProcessStatus.Idle)                            continue;                        if (wafer.WaferType == WaferType.M1)                            m1Count++;                        if (wafer.WaferType == WaferType.M2)                            m2Count++;                        if (wafer.WaferType == WaferType.P)                            pCount++;                    }                }            }            if (m1Count < pj.Monitor1Slots.Count || m2Count < pj.Monitor2Slots.Count || pCount < pj.ProductSlots.Count)                return false;            return isReady;        }        protected bool CheckAllWaferReturned(ProcessJobInfo pj, bool checkAllProcessed)        {            if (pj.SlotWafers == null || pj.SlotWafers.Count == 0 ||                pj.SlotWafers.Count != (pj.SideDummySlots != null ? pj.SideDummySlots.Count : 0) +                                       (pj.ExtraDummySlots != null ? pj.ExtraDummySlots.Count : 0) +                                       (pj.ProductSlots != null ? pj.ProductSlots.Count : 0) +                                       (pj.Monitor1Slots != null ? pj.Monitor1Slots.Count : 0) +                                      (pj.Monitor2Slots != null ? pj.Monitor2Slots.Count : 0))                return false;            var boatWafers = WaferManager.Instance.GetWafers(ModuleName.PM1).ToList();            for (int i = 0; i < pj.SlotWafers.Count; ++i)            {                if (!CarrierManager.Instance.CheckHasCarrier(pj.SlotWafers[i].Item1, 0))                {                    var isFOUPUnloaded = true;                    if (CarrierManager.Instance.CheckHasCarrier(_carrierRobot.Module, 0) && CarrierManager.Instance.GetCarrier(_carrierRobot.Module, 0).InternalModuleName == pj.SlotWafers[i].Item1)                    {                        isFOUPUnloaded = false;                    }                    foreach (var st in _lstFIMSs)                    {                        if (!st.IsOnline || !CarrierManager.Instance.CheckHasCarrier(st.Module, 0))                            continue;                        if (CarrierManager.Instance.GetCarrier(st.Module, 0).InternalModuleName == pj.SlotWafers[i].Item1)                        {                            isFOUPUnloaded = false;                            break;                        }                    }                    if (isFOUPUnloaded)                        continue;//已经unload                }                WaferInfo wafer = WaferManager.Instance.GetWafer(pj.SlotWafers[i].Item1, pj.SlotWafers[i].Item2);                if (wafer.IsEmpty)                {                    if (SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))                        return false;                    else                    {                        if (boatWafers.Any(x => x.OriginStation == (int)pj.SlotWafers[i].Item1 && x.OriginSlot == pj.SlotWafers[i].Item2 &&                        !x.IsEmpty &&                        (x.WaferType == WaferType.SD || x.WaferType == WaferType.ED)))                            continue;                        else                            return false;                    }                }                if (checkAllProcessed && CheckWaferNeedProcess(pj.SlotWafers[i].Item1, pj.SlotWafers[i].Item2))                    return false;            }            if (checkAllProcessed)            {                if (boatWafers != null)                {                    for (int i = 0; i < boatWafers.Count; i++)                    {                        var wafer = boatWafers[i];                        if (wafer.IsEmpty)                            continue;                        if (wafer.ProcessState != EnumWaferProcessStatus.Completed && wafer.ProcessState != EnumWaferProcessStatus.Failed)                        {                            return false;                        }                    }                }            }            return true;        }        public bool CheckWaferNeedProcess(ModuleName waferModule, int waferSlot, ModuleName processIn = ModuleName.System)        {            WaferInfo wafer = WaferManager.Instance.GetWafer(waferModule, waferSlot);            if (wafer.IsEmpty)                return false;            if (wafer.ProcessJob == null || wafer.ProcessJob.Sequence == null)                return false;            if (wafer.NextSequenceStep >= wafer.ProcessJob.Sequence.Steps.Count)                return false;            for (int i = wafer.NextSequenceStep; i < wafer.ProcessJob.Sequence.Steps.Count; i++)            {                if (wafer.ProcessJob.Sequence.Steps[i].StepModules                        .Contains(ModuleName.PM1) || wafer.ProcessJob.Sequence.Steps[i].StepModules                        .Contains(ModuleName.PM2))                {                    if (processIn == ModuleName.System)                        return true;                    //if (_mapLpPm[(ModuleName)wafer.OriginStation] != processIn)                    //    return false;                    if (wafer.ProcessJob.Sequence.Steps[i].StepModules                        .Contains(processIn))                        return true;                }            }            return false;        }        private List<Tuple<WaferType, int>> GetWaferTypeSlots()        {            List<Tuple<WaferType, int>> slots = new List<Tuple<WaferType, int>>();            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing);            if (pj == null)                return slots;            if (pj.SideDummySlots != null && pj.SideDummySlots.Count > 0)            {                pj.SideDummySlots.ForEach(x => slots.Add(Tuple.Create(WaferType.SD, x)));            }            if (pj.ExtraDummySlots != null && pj.ExtraDummySlots.Count > 0)            {                pj.ExtraDummySlots.ForEach(x => slots.Add(Tuple.Create(WaferType.ED, x)));            }            if (pj.ProductSlots != null && pj.ProductSlots.Count > 0)            {                pj.ProductSlots.ForEach(x => slots.Add(Tuple.Create(WaferType.P, x)));            }            if (pj.Monitor1Slots != null && pj.Monitor1Slots.Count > 0)            {                pj.Monitor1Slots.ForEach(x => slots.Add(Tuple.Create(WaferType.M1, x)));            }            if (pj.Monitor2Slots != null && pj.Monitor2Slots.Count > 0)            {                pj.Monitor2Slots.ForEach(x => slots.Add(Tuple.Create(WaferType.M2, x)));            }            return slots.OrderByDescending(x => x.Item2).ToList();        }        private string GetMappingSlot(ModuleName module)        {            string mappingSlot = string.Empty;            var wafers = WaferManager.Instance.GetWafers(module);            if (wafers != null)            {                for (int i = 0; i < wafers.Length; i++)                {                    if (!wafers[i].IsEmpty)                        mappingSlot += "1";                    else                        mappingSlot += "0";                }            }            return mappingSlot;        }        private void AdjustDischargeSlots()        {            _sideDummySlots.Clear();            _extraDummySlots.Clear();            _productSlots.Clear();            _monitor1Slots.Clear();            _monitor2Slots.Clear();            for (int i = 0; i < SC.GetValue<int>($"Boat.SlotCount"); i++)            {                if (!WaferManager.Instance.CheckHasWafer(_pm1.Module, i))                    continue;                var wafer = WaferManager.Instance.GetWafer(_pm1.Module, i);                if (wafer == null || wafer.IsEmpty)                    continue;                switch (wafer.WaferType)                {                    case WaferType.SD:                        _sideDummySlots.Add(i);                        break;                    case WaferType.ED:                        _extraDummySlots.Add(i);                        break;                    case WaferType.P:                        _productSlots.Add(i);                        break;                    case WaferType.M1:                        _monitor1Slots.Add(i);                        break;                    case WaferType.M2:                        _monitor2Slots.Add(i);                        break;                }            }            if (SC.GetStringValue("System.ChargingOrderToBoat") == "FromTopToBottom")            {                _productSlots.Sort((x, y) => x.CompareTo(y));//升序                _monitor1Slots.Sort((x, y) => x.CompareTo(y));//升序                _monitor2Slots.Sort((x, y) => x.CompareTo(y));//升序                _extraDummySlots.Sort((x, y) => x.CompareTo(y));//升序                _sideDummySlots.Sort((x, y) => x.CompareTo(y));//升序            }            else            {                _productSlots.Sort((x, y) => -x.CompareTo(y));//降序                _monitor1Slots.Sort((x, y) => -x.CompareTo(y));//降序                _monitor2Slots.Sort((x, y) => -x.CompareTo(y));//降序                _extraDummySlots.Sort((x, y) => -x.CompareTo(y));//降序                _sideDummySlots.Sort((x, y) => -x.CompareTo(y));//降序            }        }        public void CreatePj(string processName)        {            _dbCallback.PjCreated(processName);        }        public void StartPj()        {            _dbCallback.PjStart();        }        public void EndPj(string endStatus)        {            _dbCallback.PjUpdated();            _dbCallback.PjFinished(endStatus);        }        public void PjUpdated()        {            _dbCallback.PjUpdated();        }        public void PjFinished(string endStatus)        {            _dbCallback.PjFinished(endStatus);        }        #endregion Logic Check        public void CoolingSkip()        {            _isCoolingSkip = true;        }        public string GetFirstPJId()        {            return _dbCallback.GetFirstPJId();        }        public string GetFirstPJStatus()        {            if (_lstProcessJobs == null || _lstProcessJobs.Count == 0)            {                return "";            }            return _lstProcessJobs[0].ProcessingState.ToString();        }        #region FA        public bool IsCJExisted(string cjid)        {            return _lstControlJobs.Exists(x => x.Name == cjid);        }        public bool IsCJExcuting(string cjid)        {            return _lstControlJobs.Exists(x => x.Name == cjid && x.State == EnumControlJobState.Executing);        }        public bool CreateControlJob(string jobId, List<string> pjIDs, out ControlJobInfo cj)        {            cj = null;            if (_lstControlJobs.Exists(x => x.Name == jobId))            {                //EV.PostWarningLog(LogSource, $"{jobId} is already created");                return false;            }            cj = new ControlJobInfo();            cj.Name = jobId;            //cj.Module = module;            cj.LotName = jobId;            cj.LotInnerId = Guid.NewGuid();            cj.LotWafers = new List<WaferInfo>();            cj.CreateTime = DateTime.Now;            cj.SetState(EnumControlJobState.Queued);            foreach (var pjName in pjIDs)            {                var pj = _lstProcessJobs.FirstOrDefault(x => x.Name == pjName);                if (pj == null)                {                    LOG.Info($"not find {pjName} while create control job");                    continue;                }                pj.ControlJobName = jobId;                cj.ProcessJobNameList.Add(pj.Name);            }            if (!_lstControlJobs.Contains(cj))                _lstControlJobs.Add(cj);            _faCallback.JobCreated(cj);            //if ((Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule).IsExcuteIdleRecipe)            //    (Singleton<EquipmentManager>.Instance.Modules[ModuleName.PM1] as PMModule).EndIdleRecipe();            return true;        }        private bool CheckWaferStatus(List<Tuple<string, string>> useStockers, bool isNeedCheckMonitor, bool isNeedCheckP, out string reason)        {            reason = string.Empty;            if (SC.GetValue<bool>("System.IsCycleMode"))            {                foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)                {                    if (!ModuleHelper.IsStocker(module) || !useStockers.Any(x => x.Item1 == module.ToString()))                        continue;                    var wafers = WaferManager.Instance.GetWafers(module);                    if (wafers == null)                        continue;                    foreach (var wafer in wafers)                    {                        if (wafer.IsEmpty)                            continue;                        wafer.ProcessState = EnumWaferProcessStatus.Idle;                    }                }            }            else            {                foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)                {                    if (!ModuleHelper.IsStocker(module) || !useStockers.Any(x => x.Item1 == module.ToString()))                        continue;                    var wafers = WaferManager.Instance.GetWafers(module);                    if (wafers == null)                        continue;                    if (SC.GetValue<bool>("System.EnableWarningFailedWafer"))                    {                        if (wafers.ToList().Any(x => x.ProcessState == EnumWaferProcessStatus.Failed))                        {                            reason = $"{module} has failed wafers.";                            return false;                        }                    }                    foreach (var wafer in wafers)                    {                        if (wafer.IsEmpty)                            continue;                        if (wafer.ProcessState != EnumWaferProcessStatus.Idle)                        {                            if (SC.GetValue<bool>("System.EnableWarningWaferProcessState"))                            {                                if (wafer.WaferType == WaferType.P && isNeedCheckP)                                {                                    reason = $"{(ModuleName)wafer.OriginStation} {wafer.WaferType} wafer already completed process.";                                    return false;                                }                                if (isNeedCheckMonitor)                                {                                    if (wafer.WaferType == WaferType.M)                                    {                                        reason = $"{(ModuleName)wafer.OriginStation} {wafer.WaferType} wafer already completed process.";                                        return false;                                    }                                }                            }                            else                            {                                wafer.ProcessState = EnumWaferProcessStatus.Idle;                            }                        }                        if (wafer.WaferType == WaferType.ED || wafer.WaferType == WaferType.SD)                            wafer.ProcessState = EnumWaferProcessStatus.Idle;                    }                }            }            return true;        }        public bool CreateProcessJob(out ProcessJobInfo pj, params object[] objs)        {            var param = (Dictionary<string, object>)objs[0];            pj = null;            foreach (var module in Singleton<EquipmentManager>.Instance.Modules.Keys)            {                var lpModule = Singleton<EquipmentManager>.Instance.Modules[module] as LoadPortModule;                if (lpModule != null)                {                    if (lpModule.FsmState == (int)LoadPortModule.STATE.Init)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is init state");                        return false;                    }                    if (lpModule.FsmState == (int)LoadPortModule.STATE.Error)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is error state");                        return false;                    }                }                var fimsModule = Singleton<EquipmentManager>.Instance.Modules[module] as FIMSModule;                if (fimsModule != null)                {                    if (fimsModule.FsmState == (int)FIMSModule.STATE.Init)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is init state");                        return false;                    }                    if (fimsModule.FsmState == (int)FIMSModule.STATE.Error)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is error state");                        return false;                    }                }                var boatModule = Singleton<EquipmentManager>.Instance.Modules[module] as BoatModule;                if (boatModule != null)                {                    if (boatModule.FsmState == (int)BoatModule.STATE.Init)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is init state");                        return false;                    }                    if (boatModule.FsmState == (int)BoatModule.STATE.Error)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is error state");                        return false;                    }                }                var carrierRobotModule = Singleton<EquipmentManager>.Instance.Modules[module] as CarrierRobotModule;                if (carrierRobotModule != null)                {                    if (carrierRobotModule.FsmState == (int)CarrierRobotModule.STATE.Init)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is init state");                        return false;                    }                    if (carrierRobotModule.FsmState == (int)CarrierRobotModule.STATE.Error)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is error state");                        return false;                    }                }                var waferRobotModule = Singleton<EquipmentManager>.Instance.Modules[module] as WaferRobotModule;                if (waferRobotModule != null)                {                    if (waferRobotModule.FsmState == (int)WaferRobotModule.STATE.Init)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is init state");                        return false;                    }                    if (waferRobotModule.FsmState == (int)WaferRobotModule.STATE.Error)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is error state");                        return false;                    }                }                var pmModule = Singleton<EquipmentManager>.Instance.Modules[module] as PMModule;                if (pmModule != null)                {                    if (pmModule.FsmState == (int)PMModule.STATE.Init)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is init state");                        return false;                    }                    if (pmModule.FsmState == (int)PMModule.STATE.Error)                    {                        Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"{module} is error state");                        return false;                    }                }            }            if (!Singleton<EquipmentManager>.Instance.GetJobRecipeInfor((string)param["JobRecipe"], out string processRecipe, out string layoutRecipe, out int coolTime, out string reason))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set(reason);                return false;            }            if (string.IsNullOrEmpty(layoutRecipe))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set("Parameter not valid, Layout Recipe is null");                return false;            }            var path = $"{PathManager.GetRecipeDir()}\\{SC.GetStringValue("System.Recipe.SupportedChamberType")}\\Layout\\{layoutRecipe}.rcp";            if (!File.Exists(path))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Layout \"{layoutRecipe}\" not exist.");                return false;            }            path = $"{PathManager.GetRecipeDir()}\\{SC.GetStringValue("System.Recipe.SupportedChamberType")}\\Process\\{processRecipe}.rcp";            if (!File.Exists(path))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Process recipe \"{processRecipe}\" not exist.");                return false;            }            if (!RecipeParser.Parse(processRecipe, ModuleName.PM1.ToString(), out var recipeHead, out var recipeSteps, out reason, "Process"))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Load process recipe {processRecipe} failed, {reason}");                return false;            }            List<Tuple<string, string, List<int>>> useStockers = new List<Tuple<string, string, List<int>>>();            if (RecipeParser.LayoutRecipeParse(layoutRecipe, ModuleName.PM1.ToString(), out RecipeLayoutEntityNormal layoutRecipeDataNormal, out RecipeLayoutEntityExpert layoutRecipeDataExpert, out reason))            {                if (param.ContainsKey("CarrierPara"))                {                    //Stocker1,P_00000036,25,Exist,1111111111111111111111111,LP1;Stocker2,P_00000037,25,Exist,1111111111111111111111111,LP2                    var paras = param["CarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        foreach (var item in paras)                        {                            //Stocker13,,21/22/23/24/25"                            var arry = item.Split(',');                            if (arry != null && arry.Length >= 2)                            {                                var pStocker = GetModule(arry[0]) as SchedulerStocker;                                if (pStocker != null && (SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("P") || SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("None")))                                {                                    var slots = new List<int>();                                    if (arry.Length >= 3)                                    {                                        var slotParas = arry[2].Split('/').ToList();                                        if (slotParas != null && slotParas.Count > 0)                                        {                                            foreach (var slotPara in slotParas)                                            {                                                var ret = int.TryParse(slotPara, out int slot);                                                if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                {                                                    if (_lstProcessJobs.Any())                                                    {                                                        foreach (var job in _lstProcessJobs)                                                        {                                                            if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                            {                                                                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                return false;                                                            }                                                        }                                                    }                                                    slots.Add(slot - 1);                                                }                                            }                                        }                                    }                                    useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                }                            }                        }                    }                }                if (param.ContainsKey("mCarrierPara"))                {                    //Stocker14,P_00000014;Stocker15,P_00000015                    var paras = param["mCarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        if (!string.IsNullOrEmpty(paras[0]))                        {                            foreach (var item in paras)                            {                                var arry = item.Split(',');                                if (arry != null && arry.Length >= 2)                                {                                    var mStocker = GetModule(arry[0]) as SchedulerStocker;                                    if (mStocker != null && SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("M"))                                    {                                        var slots = new List<int>();                                        if (arry.Length >= 3)                                        {                                            var slotParas = arry[2].Split('/').ToList();                                            if (slotParas != null && slotParas.Count > 0)                                            {                                                foreach (var slotPara in slotParas)                                                {                                                    var ret = int.TryParse(slotPara, out int slot);                                                    if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                    {                                                        if (_lstProcessJobs.Any())                                                        {                                                            foreach (var job in _lstProcessJobs)                                                            {                                                                if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                                {                                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                    return false;                                                                }                                                            }                                                        }                                                        slots.Add(slot - 1);                                                    }                                                }                                            }                                        }                                        useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                    }                                }                            }                        }                    }                }                if (param.ContainsKey("m1CarrierPara"))                {                    //Stocker14,P_00000014;Stocker15,P_00000015                    var paras = param["m1CarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        if (!string.IsNullOrEmpty(paras[0]))                        {                            foreach (var item in paras)                            {                                var arry = item.Split(',');                                if (arry != null && arry.Length >= 2)                                {                                    var mStocker = GetModule(arry[0]) as SchedulerStocker;                                    if (mStocker != null && (SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("M1") || SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("None")))                                    {                                        var slots = new List<int>();                                        if (arry.Length >= 3)                                        {                                            var slotParas = arry[2].Split('/').ToList();                                            if (slotParas != null && slotParas.Count > 0)                                            {                                                foreach (var slotPara in slotParas)                                                {                                                    var ret = int.TryParse(slotPara, out int slot);                                                    if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                    {                                                        if (_lstProcessJobs.Any())                                                        {                                                            foreach (var job in _lstProcessJobs)                                                            {                                                                if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                                {                                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                    return false;                                                                }                                                            }                                                        }                                                        slots.Add(slot - 1);                                                    }                                                }                                            }                                        }                                        useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                    }                                }                            }                        }                    }                }                if (param.ContainsKey("m2CarrierPara"))                {                    //Stocker14,P_00000014;Stocker15,P_00000015                    var paras = param["m2CarrierPara"].ToString().Split(';');                    if (paras != null && paras.Length > 0)                    {                        if (!string.IsNullOrEmpty(paras[0]))                        {                            foreach (var item in paras)                            {                                var arry = item.Split(',');                                if (arry != null && arry.Length >= 2)                                {                                    var mStocker = GetModule(arry[0]) as SchedulerStocker;                                    if (mStocker != null && (SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("M2") || SC.GetStringValue($"System.Stocker.{arry[0]}WaferType").Contains("None")))                                    {                                        var slots = new List<int>();                                        if (arry.Length >= 3)                                        {                                            var slotParas = arry[2].Split('/').ToList();                                            if (slotParas != null && slotParas.Count > 0)                                            {                                                foreach (var slotPara in slotParas)                                                {                                                    var ret = int.TryParse(slotPara, out int slot);                                                    if (ret && slot - 1 >= 0 && !slots.Any(x => x == slot - 1))                                                    {                                                        if (_lstProcessJobs.Any())                                                        {                                                            foreach (var job in _lstProcessJobs)                                                            {                                                                if (job.Stockers.Any(x => x.Item1 == arry[0] && x.Item3.Contains(slot - 1)))                                                                {                                                                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Create job failed, already contains {arry[0]}.{slot} in previous process job");                                                                    return false;                                                                }                                                            }                                                        }                                                        slots.Add(slot - 1);                                                    }                                                }                                            }                                        }                                        useStockers.Add(Tuple.Create(arry[0], arry[1], slots));                                    }                                }                            }                        }                    }                }            }            if (Singleton<FAJobController>.Instance.FaControlState != FAControlState.OnlineRemote.ToString())                useStockers.Sort(compare);            foreach (var s in _lstSideDummyStocker)            {                if (CarrierManager.Instance.CheckHasCarrier(s.Module, 0) && CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierType == CarrierType.SD)                    useStockers.Add(Tuple.Create(s.Module.ToString(), CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierId, new List<int>()));                else                {                    bool onRobot = false;                    if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))                    {                        var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);                        if (carrier.CarrierType == CarrierType.SD &&                            carrier.InternalModuleName == s.Module)                        {                            useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                            onRobot = true;                        }                    }                    if (!onRobot)                    {                        bool inFIMS = false;                        foreach (var fims in _lstFIMSs)                        {                            if (!fims.IsOnline)                                continue;                            var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                            if (!carrier.IsEmpty && carrier.CarrierType == CarrierType.SD &&                                carrier.InternalModuleName == s.Module)                            {                                inFIMS = true;                                useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                            }                        }                        if (!inFIMS && CarrierManager.Instance.GetLocationByInternalCarrierModuleName(s.Module, CarrierType.SD) != null)                            useStockers.Add(Tuple.Create(s.Module.ToString(), "", new List<int>()));                    }                }            }            foreach (var s in _lstExtraDummyStocker)            {                if (CarrierManager.Instance.CheckHasCarrier(s.Module, 0) && CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierType == CarrierType.ED)                    useStockers.Add(Tuple.Create(s.Module.ToString(), CarrierManager.Instance.GetCarrier(s.Module, 0).CarrierId, new List<int>()));                else                {                    bool onRobot = false;                    if (CarrierManager.Instance.CheckHasCarrier(ModuleName.CarrierRobot, 0))                    {                        var carrier = CarrierManager.Instance.GetCarrier(ModuleName.CarrierRobot, 0);                        if (carrier.CarrierType == CarrierType.ED &&                            carrier.InternalModuleName == s.Module)                        {                            useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                            onRobot = true;                        }                    }                    if (!onRobot)                    {                        bool inFIMS = false;                        foreach (var fims in _lstFIMSs)                        {                            if (!fims.IsOnline)                                continue;                            var carrier = CarrierManager.Instance.GetCarrier(fims.Module, 0);                            if (!carrier.IsEmpty && carrier.CarrierType == CarrierType.ED &&                                carrier.InternalModuleName == s.Module)                            {                                inFIMS = true;                                useStockers.Add(Tuple.Create(s.Module.ToString(), carrier.CarrierId, new List<int>()));                            }                        }                        if (!inFIMS && CarrierManager.Instance.GetLocationByInternalCarrierModuleName(s.Module, CarrierType.SD) != null)                            useStockers.Add(Tuple.Create(s.Module.ToString(), "", new List<int>()));                    }                }            }            var wafers = WaferManager.Instance.GetWafers(_pm1.Module);            if (wafers.Any(x => !x.IsEmpty) && !_lstProcessJobs.Any(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused) &&                SC.GetValue<bool>("System.IsEnableDummyWaferDischarge"))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have wafer in boat");                return false;            }            if (wafers.Any(x => !x.IsEmpty && x.WaferType == WaferType.None))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have unknown wafer in boat");                return false;            }            var robotWafers = WaferManager.Instance.GetWafers(ModuleName.WaferRobot);            if (robotWafers.Any(x => !x.IsEmpty) && !_lstProcessJobs.Any(x => x.State == EnumProcessJobState.Processing || x.State == EnumProcessJobState.Paused))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have wafer in wafer robot");                return false;            }            if (robotWafers.Any(x => !x.IsEmpty && x.WaferType == WaferType.None))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have unknown wafer in wafer robot");                return false;            }            if (!CheckWaferStatus(useStockers, true, true, out reason))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set(reason);                return false;            }            //if (!CheckWaferStatus(useStockers, out reason))            //{            //    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set(reason);            //    return false;            //}            foreach (var stocker in useStockers)            {                if (CarrierManager.Instance.CheckHasCarrier($"{stocker.Item1}", 0) &&                    CarrierManager.Instance.GetCarrier($"{stocker.Item1}", 0).InternalModuleName.ToString() != stocker.Item1)                {                    Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Carrier of {stocker.Item1} is not match stocker module name {CarrierManager.Instance.GetCarrier($"{stocker.Item1}", 0).InternalModuleName}");                    return false;                }            }            bool isOnlyCharge = false;            if (param.ContainsKey("Charge"))            {                bool.TryParse(param["Charge"].ToString(), out isOnlyCharge);            }            var pjJobId = param["PJID"].ToString();            if (_lstProcessJobs.Exists(x => x.Name == pjJobId))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"Already have the same pj id={pjJobId}");                return false;            }            pj = new ProcessJobInfo();            pj.Name = pjJobId;            pj.JobRecipe = param.ContainsKey("JobRecipe") ? param["JobRecipe"].ToString() : "";            pj.ProcessRecipe = processRecipe;            pj.LayoutRecipe = layoutRecipe;            pj.ControlJobName = "";            pj.SetState(EnumProcessJobState.Queued);            pj.CreateTime = DateTime.Now;            pj.Stockers = useStockers;            pj.SlotWafers = new List<Tuple<ModuleName, int>>();            pj.ProcessingState = EnumProcessingState.WaitingMaterial;            pj.CoolTimeSec = coolTime;            //pj.IsEmptyLayout = !layoutRecipeDataExpert.Items.Any(x => x != "----");            //pj.IsOnlyCharge = isOnlyCharge;            if (!CheckChargeRule(pj, out bool waferShort, false, true, out reason))            {                Singleton<EquipmentManager>.Instance.ChargeWaferJobStartFailedWarning.Set($"layout recipe {layoutRecipe} check charge rule failed, {reason}");                return false;            }            _lstProcessJobs.Add(pj);            //_faCallback.JobCreated(pj);            return true;        }        #endregion        private bool InvokeCoolingAbort(string arg1, object[] args)        {            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing && x.ProcessingState == EnumProcessingState.Cooling);            if (pj != null)            {                pj.CoolTimeSec = 0;            }            return true;        }        private bool InvokeCoolingModify(string arg1, object[] args)        {            var pj = _lstProcessJobs.FirstOrDefault(x => x.State == EnumProcessJobState.Processing && x.ProcessingState == EnumProcessingState.Cooling);            if (pj != null && args != null && args.Length > 0)            {                float.TryParse(args[0].ToString(), out float addCoolTime);                pj.CoolTimeSec = (int)addCoolTime;            }            return true;        }        private List<string> GetBoatSlotMap()        {            var wafers = WaferManager.Instance.GetWafers(ModuleName.PM1);            if (wafers != null)            {                List<string> slotMap = new List<string>();                foreach (var wafer in wafers)                {                    if (wafer == null || wafer.IsEmpty)                    {                        slotMap.Add("");                        continue;                    }                    if (wafer.WaferType == WaferType.ED || wafer.WaferType == WaferType.SD)                        slotMap.Add("filler");                    else                        slotMap.Add(wafer.WaferID);                }                return slotMap;            }            return null;        }    }}
 |