/snippets/VS2010/Nemerle.VisualStudio/Project/NemerleLibraryManager.cs

http://github.com/xxVisorxx/nemerle · C# · 653 lines · 438 code · 134 blank · 81 comment · 104 complexity · 7465570579f49e063574b622bcfa0467 MD5 · raw file

  1. using System;
  2. using System.Collections.Generic;
  3. using System.Diagnostics;
  4. using System.IO;
  5. using System.Runtime.InteropServices;
  6. using System.Threading;
  7. using Microsoft.VisualStudio;
  8. using Microsoft.VisualStudio.Shell.Interop;
  9. using Microsoft.VisualStudio.TextManager.Interop;
  10. using Nemerle.Compiler;
  11. namespace Nemerle.VisualStudio.Project
  12. {
  13. /// <summary>
  14. /// Inplementation of the service that build the information to expose to the symbols
  15. /// navigation tools (class view or object browser) from the Nemerle files
  16. /// inside a hierarchy.
  17. /// </summary>
  18. [Guid(NemerleConstants.LibraryManagerGuidString)]
  19. public class NemerleLibraryManager : INemerleLibraryManager, IVsRunningDocTableEvents, IDisposable
  20. {
  21. /// <summary>
  22. /// Class storing the data about a parsing task on a nemerle module.
  23. /// A module in Nemerle is a source file, so here we use the file name to
  24. /// identify it.
  25. /// </summary>
  26. [DebuggerStepThrough]
  27. class LibraryTask
  28. {
  29. public LibraryTask(string fileName, string text)
  30. {
  31. _fileName = fileName;
  32. _text = text;
  33. }
  34. private string _fileName;
  35. public string FileName
  36. {
  37. get { return _fileName; }
  38. }
  39. private ModuleID _moduleID;
  40. public ModuleID ModuleID
  41. {
  42. get { return _moduleID; }
  43. set { _moduleID = value; }
  44. }
  45. private string _text;
  46. public string Text
  47. {
  48. get { return _text; }
  49. }
  50. }
  51. IServiceProvider _provider;
  52. uint _objectManagerCookie;
  53. uint _runningDocTableCookie;
  54. Library _library;
  55. Thread _parseThread;
  56. ManualResetEvent _requestPresent;
  57. ManualResetEvent _shutDownStarted;
  58. Queue<LibraryTask> _requests;
  59. Dictionary<uint, TextLineEventListener> _documents;
  60. Dictionary<IVsHierarchy, HierarchyListener> _hierarchies;
  61. Dictionary<ModuleID, LibraryNode> _files;
  62. public NemerleLibraryManager(IServiceProvider provider)
  63. {
  64. _provider = provider;
  65. _documents = new Dictionary<uint, TextLineEventListener>();
  66. _hierarchies = new Dictionary<IVsHierarchy, HierarchyListener>();
  67. _files = new Dictionary<ModuleID, LibraryNode>();
  68. _requests = new Queue<LibraryTask>();
  69. _requestPresent = new ManualResetEvent(false);
  70. _shutDownStarted = new ManualResetEvent(false);
  71. _parseThread = new Thread(ParseThread) {Name = "Parse thread"};
  72. _library = new Library(new Guid(NemerleConstants.LibraryGuidString));
  73. _library.LibraryCapabilities = (_LIB_FLAGS2)(_LIB_FLAGS.LF_PROJECT) | _LIB_FLAGS2.LF_SUPPORTSFILTERING | _LIB_FLAGS2.LF_SUPPORTSCALLBROWSER;
  74. _parseThread.Start();
  75. }
  76. void RegisterForRDTEvents()
  77. {
  78. if (0 != _runningDocTableCookie)
  79. return;
  80. IVsRunningDocumentTable rdt =
  81. _provider.GetService(typeof (SVsRunningDocumentTable)) as IVsRunningDocumentTable;
  82. if (null != rdt)
  83. // Do not throw here in case of error, simply skip the registration.
  84. rdt.AdviseRunningDocTableEvents(this, out _runningDocTableCookie);
  85. }
  86. void UnregisterRDTEvents()
  87. {
  88. if (0 == _runningDocTableCookie)
  89. return;
  90. IVsRunningDocumentTable rdt =
  91. _provider.GetService(typeof (SVsRunningDocumentTable)) as IVsRunningDocumentTable;
  92. if (null != rdt)
  93. // Do not throw in case of error.
  94. rdt.UnadviseRunningDocTableEvents(_runningDocTableCookie);
  95. _runningDocTableCookie = 0;
  96. }
  97. /// <summary>
  98. /// Hack. Based on the fact that we have only one fixed root _library. FindAllReferences
  99. /// search results are stored in the _library just before VS environment ask the _library
  100. /// for them.
  101. ///
  102. /// Õàê. Èñïîëüçóåì òî ÷òî ó íàñ îäíà ôèêñèðîâàííàÿ _library, ñîõðàíÿåì â íå¸
  103. /// óæå íàéäåííûå ðåçóëüòàòû äëÿ FindAllReferences íåïîñðåäñòâåííî ïåðåä òåì êàê
  104. /// ñðåäà VS ïîïðîñèò ó _library ýòè ðåçóëüòàòû ïîèñêà.
  105. /// </summary>
  106. public void OnFindAllReferencesDone(IVsSimpleObjectList2 findResults)
  107. {
  108. _library.OnFindAllReferencesDone(findResults);
  109. }
  110. #region IDisposable Members
  111. public void Dispose()
  112. {
  113. // Make sure that the parse thread can exit.
  114. //
  115. if (null != _shutDownStarted)
  116. _shutDownStarted.Set();
  117. if ((null != _parseThread) && _parseThread.IsAlive)
  118. {
  119. _parseThread.Join(500);
  120. if (_parseThread.IsAlive)
  121. _parseThread.Abort();
  122. _parseThread = null;
  123. }
  124. _requests.Clear();
  125. // Dispose all the listeners.
  126. //
  127. foreach (HierarchyListener listener in _hierarchies.Values)
  128. listener.Dispose();
  129. _hierarchies.Clear();
  130. foreach (TextLineEventListener textListener in _documents.Values)
  131. textListener.Dispose();
  132. _documents.Clear();
  133. // Remove this library from the object manager.
  134. //
  135. if (0 != _objectManagerCookie)
  136. {
  137. IVsObjectManager2 mgr = _provider.GetService(typeof (SVsObjectManager)) as IVsObjectManager2;
  138. if (null != mgr)
  139. mgr.UnregisterLibrary(_objectManagerCookie);
  140. _objectManagerCookie = 0;
  141. }
  142. // Unregister this object from the RDT events.
  143. //
  144. UnregisterRDTEvents();
  145. // Dispose the events used to syncronize the threads.
  146. //
  147. if (null != _requestPresent)
  148. {
  149. _requestPresent.Close();
  150. _requestPresent = null;
  151. }
  152. if (null != _shutDownStarted)
  153. {
  154. _shutDownStarted.Close();
  155. _shutDownStarted = null;
  156. }
  157. }
  158. #endregion
  159. #region INemerleLibraryManager
  160. public void RegisterHierarchy(IVsHierarchy hierarchy)
  161. {
  162. if (hierarchy == null || _hierarchies.ContainsKey(hierarchy))
  163. return;
  164. if (_objectManagerCookie == 0)
  165. {
  166. IVsObjectManager2 objManager =
  167. _provider.GetService(typeof (SVsObjectManager)) as IVsObjectManager2;
  168. if (null == objManager)
  169. return;
  170. ErrorHandler.ThrowOnFailure(
  171. objManager.RegisterSimpleLibrary(_library, out _objectManagerCookie));
  172. }
  173. HierarchyListener listener = new HierarchyListener(hierarchy);
  174. listener.ItemAdded += OnFileChanged;
  175. listener.ItemDeleted += OnDeleteFile;
  176. listener.StartListening(true);
  177. _hierarchies.Add(hierarchy, listener);
  178. RegisterForRDTEvents();
  179. }
  180. public void UnregisterHierarchy(IVsHierarchy hierarchy)
  181. {
  182. if ((null == hierarchy) || !_hierarchies.ContainsKey(hierarchy))
  183. return;
  184. HierarchyListener listener = _hierarchies[hierarchy];
  185. if (null != listener)
  186. listener.Dispose();
  187. _hierarchies.Remove(hierarchy);
  188. if (0 == _hierarchies.Count)
  189. UnregisterRDTEvents();
  190. lock (_files)
  191. {
  192. ModuleID[] keys = new ModuleID[_files.Keys.Count];
  193. _files.Keys.CopyTo(keys, 0);
  194. foreach (ModuleID id in keys)
  195. {
  196. if (hierarchy.Equals(id.Hierarchy))
  197. {
  198. _library.RemoveNode(_files[id]);
  199. _files.Remove(id);
  200. }
  201. }
  202. }
  203. // Remove the document listeners.
  204. //
  205. uint[] docKeys = new uint[_documents.Keys.Count];
  206. _documents.Keys.CopyTo(docKeys, 0);
  207. foreach (uint id in docKeys)
  208. {
  209. TextLineEventListener docListener = _documents[id];
  210. if (hierarchy.Equals(docListener.FileID.Hierarchy))
  211. {
  212. _documents.Remove(id);
  213. docListener.Dispose();
  214. }
  215. }
  216. }
  217. #endregion
  218. #region Parse Thread
  219. /// <summary>
  220. /// Main function of the parsing thread.
  221. /// This function waits on the queue of the parsing requests and build
  222. /// the parsing tree for a specific file. The resulting tree is built
  223. /// using LibraryNode objects so that it can be used inside the class
  224. /// view or object browser.
  225. /// </summary>
  226. void ParseThread()
  227. {
  228. const int waitTimeout = 500;
  229. // Define the array of events this function is interest in.
  230. //
  231. WaitHandle[] eventsToWait = new WaitHandle[] { _requestPresent, _shutDownStarted };
  232. // Execute the tasks.
  233. //
  234. while (true)
  235. {
  236. // Wait for a task or a shutdown request.
  237. //
  238. int waitResult = WaitHandle.WaitAny(eventsToWait, waitTimeout, false);
  239. if (1 == waitResult)
  240. // The shutdown of this component is started, so exit the thread.
  241. return;
  242. LibraryTask task = null;
  243. lock (_requests)
  244. {
  245. if (_requests.Count != 0)
  246. task = _requests.Dequeue();
  247. if (_requests.Count == 0)
  248. _requestPresent.Reset();
  249. }
  250. if (null == task)
  251. continue;
  252. ScopeNode scope = null;
  253. if (task.Text == null)
  254. {
  255. if (File.Exists(task.FileName))
  256. {
  257. Debug.WriteLine("Parse request (no text): " + task.FileName + " " + task.ModuleID);
  258. return;
  259. }
  260. }
  261. else
  262. {
  263. Debug.WriteLine("Parse request: " + task.FileName + " " + task.ModuleID);
  264. return;
  265. }
  266. LibraryNode module = new LibraryNode(
  267. Path.GetFileName(task.FileName),
  268. LibraryNode.LibraryNodeType.PhysicalContainer);
  269. CreateModuleTree(module, module, scope, "", task.ModuleID);
  270. if (task.ModuleID != null)
  271. {
  272. LibraryNode previousItem;
  273. lock (_files)
  274. if (_files.TryGetValue(task.ModuleID, out previousItem))
  275. _files.Remove(task.ModuleID);
  276. _library.RemoveNode(previousItem);
  277. }
  278. _library.AddNode(module);
  279. if (task.ModuleID != null)
  280. lock (_files)
  281. _files.Add(task.ModuleID, module);
  282. }
  283. }
  284. void CreateModuleTree(
  285. LibraryNode root,
  286. LibraryNode current,
  287. ScopeNode scope,
  288. string namePrefix,
  289. ModuleID moduleId)
  290. {
  291. if ((null == root) || (null == scope) || (null == scope.NestedScopes))
  292. return;
  293. foreach (ScopeNode subItem in scope.NestedScopes)
  294. {
  295. NemerleLibraryNode newNode = new NemerleLibraryNode(
  296. subItem, namePrefix, moduleId.Hierarchy, moduleId.ItemID);
  297. string newNamePrefix = namePrefix;
  298. // The classes are always added to the root node, the functions to the current node.
  299. //
  300. if ((newNode.NodeType & LibraryNode.LibraryNodeType.Members) != LibraryNode.LibraryNodeType.None)
  301. {
  302. current.AddNode(newNode);
  303. }
  304. else if ((newNode.NodeType & LibraryNode.LibraryNodeType.Classes) != LibraryNode.LibraryNodeType.None)
  305. {
  306. // Classes are always added to the root.
  307. //
  308. root.AddNode(newNode);
  309. newNamePrefix = newNode.Name + ".";
  310. }
  311. // Now use recursion to get the other types.
  312. //
  313. CreateModuleTree(root, newNode, subItem, newNamePrefix, moduleId);
  314. }
  315. }
  316. #endregion
  317. void CreateParseRequest(string file, string text, ModuleID id)
  318. {
  319. LibraryTask task = new LibraryTask(file, text);
  320. task.ModuleID = id;
  321. lock (_requests)
  322. _requests.Enqueue(task);
  323. _requestPresent.Set();
  324. }
  325. #region Hierarchy Events
  326. void OnFileChanged(object sender, HierarchyEventArgs args)
  327. {
  328. IVsHierarchy hierarchy = sender as IVsHierarchy;
  329. if (null == hierarchy)
  330. return;
  331. string fileText = null;
  332. if (null != args.TextBuffer)
  333. {
  334. int lastLine;
  335. int lastIndex;
  336. int hr = args.TextBuffer.GetLastLineIndex(out lastLine, out lastIndex);
  337. if (ErrorHandler.Failed(hr))
  338. return;
  339. hr = args.TextBuffer.GetLineText(0, 0, lastLine, lastIndex, out fileText);
  340. if (ErrorHandler.Failed(hr))
  341. return;
  342. var projectInfo = ProjectInfo.FindProject(hierarchy);
  343. if (null != projectInfo)
  344. {
  345. int fileIndex = Nemerle.Compiler.Location.GetFileIndex(args.FileName);
  346. projectInfo.Engine.NotifySourceChanged(new StringSource(fileIndex, fileText));
  347. }
  348. }
  349. CreateParseRequest(args.FileName, fileText, new ModuleID(hierarchy, args.ItemID));
  350. }
  351. void OnDeleteFile(object sender, HierarchyEventArgs args)
  352. {
  353. IVsHierarchy hierarchy = sender as IVsHierarchy;
  354. if (null == hierarchy)
  355. return;
  356. ModuleID id = new ModuleID(hierarchy, args.ItemID);
  357. LibraryNode node;
  358. lock (_files)
  359. if (_files.TryGetValue(id, out node))
  360. _files.Remove(id);
  361. if (null != node)
  362. _library.RemoveNode(node);
  363. }
  364. #endregion
  365. #region IVsRunningDocTableEvents Members
  366. public int OnAfterAttributeChange(uint docCookie, uint grfAttribs)
  367. {
  368. if ((grfAttribs & (uint)(__VSRDTATTRIB.RDTA_MkDocument)) == (uint)__VSRDTATTRIB.RDTA_MkDocument)
  369. {
  370. IVsRunningDocumentTable rdt =
  371. _provider.GetService(typeof (SVsRunningDocumentTable)) as IVsRunningDocumentTable;
  372. if (rdt != null)
  373. {
  374. uint flags, readLocks, editLocks, itemid;
  375. IVsHierarchy hier;
  376. IntPtr docData = IntPtr.Zero;
  377. string moniker;
  378. try
  379. {
  380. ErrorHandler.Failed(rdt.GetDocumentInfo(
  381. docCookie,
  382. out flags,
  383. out readLocks,
  384. out editLocks,
  385. out moniker,
  386. out hier,
  387. out itemid,
  388. out docData));
  389. TextLineEventListener listner;
  390. if (_documents.TryGetValue(docCookie, out listner))
  391. listner.FileName = moniker;
  392. }
  393. finally
  394. {
  395. if (IntPtr.Zero != docData)
  396. Marshal.Release(docData);
  397. }
  398. }
  399. }
  400. return VSConstants.S_OK;
  401. }
  402. public int OnAfterDocumentWindowHide(uint docCookie, IVsWindowFrame pFrame)
  403. {
  404. return VSConstants.S_OK;
  405. }
  406. public int OnAfterFirstDocumentLock(
  407. uint docCookie,
  408. uint dwRDTLockType,
  409. uint dwReadLocksRemaining,
  410. uint dwEditLocksRemaining)
  411. {
  412. return VSConstants.S_OK;
  413. }
  414. public int OnAfterSave(uint docCookie)
  415. {
  416. return VSConstants.S_OK;
  417. }
  418. public int OnBeforeDocumentWindowShow(uint docCookie, int fFirstShow, IVsWindowFrame pFrame)
  419. {
  420. // Check if this document is in the list of the documents.
  421. //
  422. if (_documents.ContainsKey(docCookie))
  423. return VSConstants.S_OK;
  424. // Get the information about this document from the RDT.
  425. //
  426. IVsRunningDocumentTable rdt =
  427. _provider.GetService(typeof (SVsRunningDocumentTable)) as IVsRunningDocumentTable;
  428. if (null != rdt)
  429. {
  430. // Note that here we don't want to throw in case of error.
  431. uint flags;
  432. uint readLocks;
  433. uint writeLoks;
  434. string documentMoniker;
  435. IVsHierarchy hierarchy;
  436. uint itemId;
  437. IntPtr unkDocData;
  438. int hr = rdt.GetDocumentInfo(
  439. docCookie,
  440. out flags,
  441. out readLocks,
  442. out writeLoks,
  443. out documentMoniker,
  444. out hierarchy,
  445. out itemId,
  446. out unkDocData);
  447. try
  448. {
  449. if (ErrorHandler.Failed(hr) || (IntPtr.Zero == unkDocData))
  450. return VSConstants.S_OK;
  451. // Check if the herarchy is one of the hierarchies this service is monitoring.
  452. //
  453. if (!_hierarchies.ContainsKey(hierarchy))
  454. // This hierarchy is not monitored, we can exit now.
  455. return VSConstants.S_OK;
  456. // Check the extension of the file to see if a listener is required.
  457. //
  458. string extension = Path.GetExtension(documentMoniker);
  459. if (string.Compare(extension, NemerleConstants.FileExtension, StringComparison.OrdinalIgnoreCase) != 0)
  460. return VSConstants.S_OK;
  461. // Create the module id for this document.
  462. //
  463. ModuleID docId = new ModuleID(hierarchy, itemId);
  464. // Try to get the text buffer.
  465. //
  466. IVsTextLines buffer = Marshal.GetObjectForIUnknown(unkDocData) as IVsTextLines;
  467. // Create the listener.
  468. //
  469. TextLineEventListener listener = new TextLineEventListener(buffer, documentMoniker, docId);
  470. // Set the event handler for the change event. Note that there is no
  471. // difference between the AddFile and FileChanged operation, so we
  472. // can use the same handler.
  473. //
  474. listener.OnFileChanged += OnFileChanged;
  475. // Add the listener to the dictionary, so we will not create it anymore.
  476. //
  477. _documents.Add(docCookie, listener);
  478. }
  479. finally
  480. {
  481. if (IntPtr.Zero != unkDocData)
  482. Marshal.Release(unkDocData);
  483. }
  484. }
  485. // Always return success.
  486. //
  487. return VSConstants.S_OK;
  488. }
  489. public int OnBeforeLastDocumentUnlock(
  490. uint docCookie,
  491. uint dwRDTLockType,
  492. uint dwReadLocksRemaining,
  493. uint dwEditLocksRemaining)
  494. {
  495. if ((0 != dwEditLocksRemaining) || (0 != dwReadLocksRemaining))
  496. return VSConstants.S_OK;
  497. TextLineEventListener listener;
  498. if (!_documents.TryGetValue(docCookie, out listener) || (null == listener))
  499. return VSConstants.S_OK;
  500. using (listener)
  501. {
  502. _documents.Remove(docCookie);
  503. // Now make sure that the information about this file are up to date
  504. // (e.g. it is possible that Class View shows something strange if the
  505. // file was closed without saving the changes).
  506. //
  507. HierarchyEventArgs args = new HierarchyEventArgs(listener.FileID.ItemID, listener.FileName);
  508. OnFileChanged(listener.FileID.Hierarchy, args);
  509. }
  510. return VSConstants.S_OK;
  511. }
  512. #endregion
  513. public void OnIdle()
  514. {
  515. foreach (TextLineEventListener listener in _documents.Values)
  516. listener.OnIdle();
  517. }
  518. }
  519. }