IossWidReader.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532
  1. using Aitex.Common.Util;
  2. using Aitex.Core.Common;
  3. using Aitex.Core.RT.Event;
  4. using Aitex.Core.RT.Log;
  5. using Aitex.Core.RT.SCCore;
  6. using MECF.Framework.Common.Communications;
  7. using MECF.Framework.Common.Equipment;
  8. using MECF.Framework.Common.SubstrateTrackings;
  9. using MECF.Framework.RT.EquipmentLibrary.HardwareUnits.OcrReaders;
  10. using System;
  11. using System.Collections.Generic;
  12. using System.IO;
  13. using System.Linq;
  14. using System.Text;
  15. using System.Text.RegularExpressions;
  16. using System.Threading.Tasks;
  17. namespace MECF.Framework.RT.EquipmentLibrary.HardwareUnits.OcrReaders.IOSS
  18. {
  19. public class IossWidReader: OCRReaderBaseDevice, IConnection
  20. {
  21. public IossWidReader(string module,string name,string scRoot):base(module,name)
  22. {
  23. _scRoot = scRoot;
  24. InitIoss();
  25. CheckToPostMessage((int)OCRReaderMsg.StartInit, null);
  26. }
  27. private void InitIoss()
  28. {
  29. JobFileList = new List<string>();
  30. lib = new Wid110Lib();
  31. _enableLog = SC.GetValue<bool>($"{_scRoot}.{Name}.EnableLogMessage");
  32. if (!isValidIP(Address))
  33. {
  34. EV.PostAlarmLog("System", $"Invalid IP address for WIDReader:{Name}");
  35. return;
  36. }
  37. if (!lib.FInit(Address))
  38. {
  39. EV.PostAlarmLog("System", $"WIDReader:{Name} initialize failed");
  40. int eno = Wid110LibConst.ecNotInit;
  41. eno = lib.FGetLastError();
  42. if (lib.isException())
  43. LOG.Write("getLastError(): EXCEPTION\r\n" + lib.getLastExcp());
  44. LOG.Write("getLastError(): return " + eno);
  45. return;
  46. }
  47. var fileInfo = new FileInfo(PathManager.GetDirectory($"Logs\\{Name}"));
  48. if (fileInfo.Directory != null && !fileInfo.Directory.Exists)
  49. fileInfo.Directory.Create();
  50. if (fileInfo.Directory != null)
  51. _imageStorePath = fileInfo.FullName;
  52. EV.PostInfoLog("System", $"WIDReader:{Name} initialize successful");
  53. IsConnected = true;
  54. }
  55. private string _imageStorePath;
  56. private bool isValidIP(string ip)
  57. {
  58. char[] sep = { '.' };
  59. string[] adr = ip.Split(sep);
  60. try
  61. {
  62. return (isValidIPPart(Convert.ToInt32(adr[0]))
  63. && isValidIPPart(Convert.ToInt32(adr[1]))
  64. && isValidIPPart(Convert.ToInt32(adr[2]))
  65. && isValidIPPart(Convert.ToInt32(adr[3])));
  66. }
  67. catch (Exception e)
  68. {
  69. LOG.Write("isValidIP( " + ip + " )\r\n"
  70. + "ERROR: Invalid IP Address\r\n"
  71. + e.ToString());
  72. return false;
  73. }
  74. }
  75. private bool isValidIPPart(int adr)
  76. {
  77. return (adr >= 0 && adr <= 255);
  78. }
  79. private Wid110Lib lib;
  80. private string _scRoot;
  81. public string Address => SC.GetStringValue($"{_scRoot}.{Name}.Address");
  82. private bool _enableLog;
  83. public bool IsConnected { get; set; }
  84. private bool _isNeedSavePicture
  85. {
  86. get
  87. {
  88. if (SC.ContainsItem($"{_scRoot}.{Name}.SavePicture"))
  89. return SC.GetValue<bool>($"{_scRoot}.{Name}.SavePicture");
  90. return false;
  91. }
  92. }
  93. protected override bool fStartSavePicture(object[] param)
  94. {
  95. return true;
  96. }
  97. protected override bool fMonitorSavingPicture(object[] param)
  98. {
  99. IsBusy = false;
  100. GetProcessImage();
  101. return true;
  102. }
  103. protected override bool fStartReset(object[] param)
  104. {
  105. IsBusy = false;
  106. _enableLog = SC.GetValue<bool>($"{_scRoot}.{Name}.EnableLogMessage");
  107. if (!isValidIP(Address))
  108. {
  109. EV.PostAlarmLog("System", $"Invalid IP address for WIDReader:{Name}");
  110. return false;
  111. }
  112. if (!lib.FInit(Address))
  113. {
  114. EV.PostAlarmLog("System", $"WIDReader:{Name} initialize address:{Address} failed");
  115. int eno = Wid110LibConst.ecNotInit;
  116. eno = lib.FGetLastError();
  117. if (lib.isException())
  118. LOG.Write("getLastError(): EXCEPTION\r\n" + lib.getLastExcp());
  119. LOG.Write("getLastError(): return " + eno);
  120. return false;
  121. }
  122. EV.PostInfoLog("System", $"WIDReader:{Name} initialize successful");
  123. IsConnected = true;
  124. return true; ;
  125. }
  126. protected override bool fStartInit(object[] param)
  127. {
  128. IsBusy = false;
  129. return true;
  130. }
  131. protected override bool fMonitorInit(object[] param)
  132. {
  133. IsBusy = false;
  134. return true;
  135. }
  136. protected override bool fStartSetParameter(object[] param)
  137. {
  138. return true ;
  139. }
  140. private string m_readparameter;
  141. protected override bool fStartReadParameter(object[] param)
  142. {
  143. m_readparameter = param[0].ToString();
  144. switch (m_readparameter)
  145. {
  146. case "JobList":
  147. if (SC.ContainsItem($"{ _scRoot}.{ Name}.DirectoryForJob") && !string.IsNullOrEmpty(SC.GetStringValue($"{ _scRoot}.{ Name}.DirectoryForJob")))
  148. {
  149. string jobpath = SC.GetStringValue($"{ _scRoot}.{ Name}.DirectoryForJob");
  150. string[] files = Directory.GetFiles(jobpath);
  151. JobFileList = new List<string>();
  152. JobFileList.Clear();
  153. foreach (string strfile in files)
  154. {
  155. FileInfo f = new FileInfo(strfile);
  156. if (f.Extension == ".job")
  157. {
  158. JobFileList.Add(f.Name);
  159. }
  160. }
  161. }
  162. break;
  163. default:
  164. break;
  165. }
  166. return true; ;
  167. }
  168. protected override bool fMonitorReadParameter(object[] param)
  169. {
  170. IsBusy = false;
  171. return true;
  172. }
  173. protected override bool fStartReadWaferID(object[] param)
  174. {
  175. //string job = param[0].ToString();
  176. //int lasermarkindex = (int)param[1];
  177. //IsReadLaserMark1 = lasermarkindex == 0;
  178. //string path = SC.GetStringValue($"{ _scRoot}.{ Name}.DirectoryForJob");
  179. //if(!lib.FLoadRecipes(path + "\\" + job))
  180. //{
  181. // string le = lib.FGetErrorDescription(lib.FGetLastError());
  182. // OnError("Load job failed:"+le);
  183. // return false;
  184. //}
  185. //if (!lib.FProcessRead())
  186. //{
  187. // string le = lib.FGetErrorDescription(lib.FGetLastError());
  188. // OnError("Read laser mark failed:" +le);
  189. // return false;
  190. //}
  191. return true;
  192. //return lib.FGetWaferId();
  193. }
  194. protected override bool fMonitorReading(object[] param)
  195. {
  196. IsBusy = false;
  197. string job = CurrentParamter[0].ToString();
  198. int lasermarkindex = (int)CurrentParamter[1];
  199. IsReadLaserMark1 = lasermarkindex == 0;
  200. string path = SC.GetStringValue($"{ _scRoot}.{ Name}.DirectoryForJob");
  201. if(!lib.FLoadRecipes(path + "\\" + job))
  202. {
  203. if (!lib.FInit(Address))
  204. {
  205. EV.PostAlarmLog("System", $"WIDReader:{Name} initialize address:{Address} failed");
  206. int eno = Wid110LibConst.ecNotInit;
  207. eno = lib.FGetLastError();
  208. if (lib.isException())
  209. LOG.Write("getLastError(): EXCEPTION\r\n" + lib.getLastExcp());
  210. LOG.Write("getLastError(): return " + eno);
  211. string le = lib.FGetErrorDescription(lib.FGetLastError());
  212. OnError("Load job failed:" + le);
  213. ReadOK = false;
  214. return false;
  215. }
  216. else if(!lib.FLoadRecipes(path + "\\" + job))
  217. {
  218. string le = lib.FGetErrorDescription(lib.FGetLastError());
  219. OnError("Load job failed:" + le);
  220. ReadOK = false;
  221. return false;
  222. }
  223. }
  224. if (!lib.FProcessRead())
  225. {
  226. string le = lib.FGetErrorDescription(lib.FGetLastError());
  227. OnError("Read laser mark failed:" + le);
  228. ReadOK = false;
  229. return false;
  230. }
  231. ReadOK = true;
  232. CurrentLaserMark = lib.FGetWaferId().Replace("READ:","").Trim();
  233. if(CurrentLaserMark.Length > KeepLaserMarkCharatersNumber)
  234. {
  235. CurrentLaserMark = CurrentLaserMark.Substring(0, KeepLaserMarkCharatersNumber);
  236. }
  237. CurrentLaserMarkScore = Convert.ToSingle(lib.FGetCodeQualityOCR());
  238. if (IsReadLaserMark1)
  239. {
  240. LaserMark1 = CurrentLaserMark;
  241. LaserMark1Score = CurrentLaserMarkScore.ToString();
  242. LaserMark1ReadTime = lib.FGetCodeTime().ToString();
  243. LaserMark1ReadResult = $"ID:{LaserMark1},Score:{LaserMark1Score},Read time:{LaserMark1ReadTime}";
  244. }
  245. else
  246. {
  247. LaserMark2 = CurrentLaserMark;
  248. LaserMark2Score = CurrentLaserMarkScore.ToString();
  249. LaserMark2ReadTime = lib.FGetCodeTime().ToString();
  250. LaserMark2ReadResult = $"ID:{LaserMark2},Score:{LaserMark2Score},Read time:{LaserMark2ReadTime}";
  251. }
  252. if(_isNeedSavePicture)
  253. {
  254. GetProcessImage();
  255. }
  256. IsBusy = false;
  257. return true;
  258. }
  259. private void GetProcessImage(int bestOrAll=0)
  260. {
  261. try
  262. {
  263. if (!lib.FIsInitialized())
  264. {
  265. string errMsg;
  266. if (lib.CheckError(out errMsg))
  267. {
  268. OnError("ERROR: " + errMsg);
  269. }
  270. return;
  271. }
  272. if (!lib.FProcessGetImage(CurrentImageFileName, bestOrAll))
  273. {
  274. if (lib.getErrno() == Wid110LibConst.ecNoMoreImg)
  275. {
  276. EV.PostInfoLog("WIDReader", "No images retrieved after last process trigger.");
  277. }
  278. string errMsg;
  279. if (lib.CheckError(out errMsg))
  280. {
  281. EV.PostWarningLog(Name, "ERROR: " + errMsg);
  282. }
  283. }
  284. else
  285. {
  286. if (CurrentLaserMarkScore < ScoreThresholdToSaveImage)
  287. {
  288. if(SavePictureOption == PictureOptionEnum.All || SavePictureOption == PictureOptionEnum.Failed)
  289. SaveCurrentImageToFailedImage();
  290. }
  291. else
  292. {
  293. if (SavePictureOption == PictureOptionEnum.All || SavePictureOption == PictureOptionEnum.Success)
  294. SaveCurrentImage();
  295. }
  296. }
  297. DeleteEarlyImageFile(PicturesQuanlityLimit);
  298. }
  299. catch(Exception ex)
  300. {
  301. LOG.Write(ex);
  302. }
  303. }
  304. private string _currentWaferID => WaferManager.Instance.GetWafer(InstalledModule, 0).WaferID ?? "";
  305. public override string[] GetJobFileList()
  306. {
  307. return JobFileList.ToArray();
  308. }
  309. public bool Connect()
  310. {
  311. return true;
  312. }
  313. public bool Disconnect()
  314. {
  315. return true;
  316. }
  317. public override void Terminate()
  318. {
  319. try
  320. {
  321. if (!lib.FExit())
  322. {
  323. LOG.Write(lib.FGetErrorDescription(lib.FGetLastError()));
  324. }
  325. }
  326. catch(Exception ex)
  327. {
  328. LOG.Write(ex);
  329. }
  330. }
  331. private bool SaveCurrentImageToFailedImage(string wid = "")
  332. {
  333. try
  334. {
  335. string waferID = WaferManager.Instance.GetWaferID(ModuleName.Aligner, 0) ?? "";
  336. if (wid != "") waferID = wid;
  337. if (File.Exists(CurrentImageFileName))
  338. {
  339. string[] temps = CurrentImageFileName.Split('\\');
  340. string pathname = CurrentImageFileName.Replace(temps.LastOrDefault(), "") + "FailedPictures\\"
  341. + DateTime.Now.ToString("yyyy-MM-dd");
  342. if (!Directory.Exists(pathname))
  343. {
  344. Directory.CreateDirectory(pathname);
  345. }
  346. string failedImage = pathname + "\\" + "Failed_" + waferID + "_" + string.Format("{0:yyyyMMddHHmmss}", DateTime.Now) + ".bmp";
  347. File.Copy(CurrentImageFileName, failedImage, true);
  348. LOG.Write($"Read success image file for {waferID} was save as {failedImage}.");
  349. }
  350. return true;
  351. }
  352. catch (Exception ex)
  353. {
  354. LOG.Write(ex);
  355. return false;
  356. }
  357. }
  358. private bool SaveCurrentImage(string wid = "")
  359. {
  360. try
  361. {
  362. string waferID = WaferManager.Instance.GetWaferID(ModuleName.Aligner, 0) ?? "";
  363. if (wid != "") waferID = wid;
  364. if (File.Exists(CurrentImageFileName))
  365. {
  366. string[] temps = CurrentImageFileName.Split('\\');
  367. string pathname = CurrentImageFileName.Replace(temps.LastOrDefault(), "") + "SuccessPictures\\"
  368. + DateTime.Now.ToString("yyyy-MM-dd");
  369. if (!Directory.Exists(pathname))
  370. {
  371. Directory.CreateDirectory(pathname);
  372. }
  373. string successimage = pathname + "\\" + "Success_" +
  374. waferID + "_" + string.Format("{0:yyyyMMddHHmmss}", DateTime.Now) + ".bmp";
  375. File.Copy(CurrentImageFileName, successimage, true);
  376. LOG.Write($"Read success image file for {waferID} was save as {successimage}.");
  377. return true;
  378. }
  379. return true;
  380. }
  381. catch (Exception ex)
  382. {
  383. LOG.Write(ex);
  384. return false;
  385. }
  386. }
  387. private void DeleteEarlyImageFile(int fileCount)
  388. {
  389. try
  390. {
  391. string[] temps = CurrentImageFileName.Split('\\');
  392. string imagepath = CurrentImageFileName.Replace("\\" + temps.LastOrDefault(), "");
  393. if (string.IsNullOrEmpty(imagepath)) return;
  394. string[] fpath = Directory.GetFiles(imagepath, "*.bmp", SearchOption.AllDirectories);
  395. Dictionary<string, DateTime> fCreateDate = new Dictionary<string, DateTime>();
  396. for (int i = 0; i < fpath.Length; i++)
  397. {
  398. FileInfo fi = new FileInfo(fpath[i]);
  399. fCreateDate[fpath[i]] = fi.CreationTime;
  400. }
  401. fCreateDate = fCreateDate.OrderBy(f => f.Value).ToDictionary(f => f.Key, f => f.Value);
  402. int fCount = fCreateDate.Count;
  403. while (fCount > fileCount)
  404. {
  405. string strFile = fCreateDate.First().Key;
  406. if (strFile == CurrentImageFileName)
  407. {
  408. fCreateDate.Remove(strFile);
  409. fCount = fCreateDate.Count;
  410. continue;
  411. }
  412. File.Delete(strFile);
  413. fCreateDate.Remove(strFile);
  414. fCount = fCreateDate.Count;
  415. LOG.Write($"Success to delete file:{strFile}");
  416. }
  417. DeleteEmptyDirectory(imagepath);
  418. LOG.Write($"Complete to delete file");
  419. }
  420. catch (Exception ex)
  421. {
  422. }
  423. }
  424. public static void DeleteEmptyDirectory(String storagepath)
  425. {
  426. DirectoryInfo dir = new DirectoryInfo(storagepath);
  427. DirectoryInfo[] subdirs = dir.GetDirectories("*.*", SearchOption.AllDirectories);
  428. foreach (DirectoryInfo subdir in subdirs)
  429. {
  430. FileSystemInfo[] subFiles = subdir.GetFileSystemInfos();
  431. if (subFiles.Count() == 0)
  432. {
  433. subdir.Delete();
  434. }
  435. }
  436. }
  437. }
  438. }