|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259 |
- //
- // Lol Engine - VsLol add-in for Visual Studio
- //
- // Copyright: (c) 2010-2013 Sam Hocevar <sam@hocevar.net>
- // This program is free software; you can redistribute it and/or
- // modify it under the terms of the Do What The Fuck You Want To
- // Public License, Version 2, as published by Sam Hocevar. See
- // http://www.wtfpl.net/ for more details.
- //
-
- using System;
- using System.Collections;
- using System.Collections.Generic;
- using System.ComponentModel.Design;
- using System.Diagnostics;
- using System.Globalization;
- using System.IO;
- using System.Runtime.InteropServices;
- using System.Text;
-
- using EnvDTE;
-
- using Microsoft.VisualStudio.Shell;
- using Microsoft.VisualStudio.Shell.Interop;
-
- using VSConstants = Microsoft.VisualStudio.VSConstants;
-
- namespace lol
- {
-
- internal class MenuGenerateCompilers : OleMenuCommand
- {
- public MenuGenerateCompilers(ServiceProvider sp, CommandID id) :
- base(new EventHandler(ClickCallback), id, VsLol.ResourceManager.GetString("GenerateCompilersText"))
- {
- this.sp = sp;
- this.projects = new List<Project>();
- this.BeforeQueryStatus += new EventHandler(OnBeforeQueryStatus);
- }
-
- private void OnBeforeQueryStatus(object sender, EventArgs e)
- {
- projects.Clear();
-
- var cmd = sender as OleMenuCommand;
- if (cmd == null)
- return;
-
- IVsMonitorSelection monitorSelection = sp.GetService(typeof(IVsMonitorSelection)) as IVsMonitorSelection;
- if (monitorSelection == null)
- return;
-
- IntPtr hier = IntPtr.Zero;
- UInt32 itemid;
- IVsMultiItemSelect multiitem = null;
- IntPtr container = IntPtr.Zero;
-
- try
- {
- monitorSelection.GetCurrentSelection(out hier, out itemid, out multiitem, out container);
-
- /* Bail out if nothing is selected */
- if (itemid != VSConstants.VSITEMID_SELECTION && itemid != VSConstants.VSITEMID_NIL)
- {
- if (hier == IntPtr.Zero)
- {
- /* FIXME: parse the whole solution */
- }
- else
- {
- object project = null;
-
- IVsHierarchy hierarchy = (IVsHierarchy)Marshal.GetObjectForIUnknown(hier);
- hierarchy.GetProperty(VSConstants.VSITEMID_ROOT, (int)__VSHPROPID.VSHPROPID_ExtObject, out project);
- projects.Add(project as Project);
- }
- }
- }
- finally
- {
- if (hier != IntPtr.Zero)
- Marshal.Release(hier);
-
- if (container != IntPtr.Zero)
- Marshal.Release(container);
- }
-
- // If there are .l or .y files in this project, display the context menu
- Visible = false;
- foreach (Project project in projects)
- foreach (ProjectItem item in ParseProjectItems(project))
- {
- if (item.Name.EndsWith("-scanner.l", StringComparison.CurrentCultureIgnoreCase)
- || item.Name.EndsWith("-parser.y", StringComparison.CurrentCultureIgnoreCase))
- Visible = true;
- }
- }
-
- private static void ClickCallback(object sender, EventArgs args)
- {
- MenuGenerateCompilers cmd = sender as MenuGenerateCompilers;
- if (cmd == null)
- return;
-
- Logger.Clear();
- Logger.Info("------ Build started: Generating Compilers ------\n");
-
- int scanner_count = 0, parser_count = 0, error_count = 0;
-
- foreach (Project project in cmd.projects)
- {
- Logger.Info("Project " + project.Name + "\n");
-
- string project_path = Path.GetDirectoryName(project.FullName);
-
- /* FIXME: find this using the solution globals! */
- string external_path = project_path;
- for (int i = 0; i < 10; ++i)
- {
- external_path += "\\..";
- if (Directory.Exists(external_path + "\\external"))
- break;
- }
-
- /* FIXME: do not hardcode shit! */
- string flex_path = external_path + "\\external\\flex-2.5.35";
- string bison_path = external_path + "\\external\\bison-2.4.2";
-
- /* Workaround for an MSYS bug. If these directories don't
- * exist, fork() will fail. Yeah, wtf. */
- try
- {
- Directory.CreateDirectory(flex_path + "\\etc");
- Directory.CreateDirectory(bison_path + "\\etc");
- }
- catch (Exception e) { }
-
- // Run flex on all the .l files
- foreach (ProjectItem item in ParseProjectItems(project))
- {
- string filename = item.get_FileNames(0);
-
- if (filename.StartsWith(project_path + "\\"))
- {
- filename = filename.Substring(project_path.Length + 1);
- filename = filename.Replace("\\", "/");
- }
-
- if (item.Name.EndsWith("-scanner.l"))
- {
- Logger.Info("flex.exe " + filename + "\n");
-
- string basename = Path.GetFileName(filename.Substring(0, filename.LastIndexOf("-scanner.l")));
- if (!cmd.Run(project_path,
- flex_path + "\\bin\\flex.exe",
- "-v -o "
- + "generated/" + basename + "-scanner.cpp "
- + filename,
- ""))
- ++error_count;
-
- ++scanner_count;
- }
-
- if (item.Name.EndsWith("-parser.y"))
- {
- Logger.Info("bison.exe " + filename + "\n");
-
- string basename = Path.GetFileName(filename.Substring(0, filename.LastIndexOf("-parser.y")));
- if (!cmd.Run(project_path,
- bison_path + "\\bin\\bison.exe",
- "-v -o "
- + "generated/" + basename + "-parser.cpp "
- + "--defines=generated/" + basename + "-parser.h "
- + "-d "
- + "-b "
- + "generated/" + basename + " "
- + filename,
- "BISON_PKGDATADIR=" + bison_path + "\\share\\bison"))
- ++error_count;
-
- ++parser_count;
- }
- }
- }
-
- Logger.Info(string.Format("========== Done: {0} scanner(s), {1} parser(s), {2} error(s) ==========\n",
- scanner_count, parser_count, error_count));
- }
-
- bool Run(string directory, string executable, string arguments, string env)
- {
- System.Diagnostics.Process p = new System.Diagnostics.Process();
- p.StartInfo.FileName = executable;
- p.StartInfo.Arguments = arguments;
- foreach (string s in env.Split(new char[]{'\n'}, StringSplitOptions.RemoveEmptyEntries))
- {
- int i = s.IndexOf("=");
- if (i > 0 && i < s.Length - 1)
- p.StartInfo.EnvironmentVariables[s.Substring(0, i - 1)] = s.Substring(i + 1);
- }
- p.StartInfo.WorkingDirectory = directory;
- p.StartInfo.CreateNoWindow = true;
- p.StartInfo.WindowStyle = ProcessWindowStyle.Hidden;
- p.StartInfo.RedirectStandardError = true;
- p.StartInfo.RedirectStandardOutput = true;
- p.StartInfo.RedirectStandardInput = true;
- p.StartInfo.UseShellExecute = false;
-
- try
- {
- p.Start();
- string output = p.StandardError.ReadToEnd()
- + p.StandardOutput.ReadToEnd();
- p.WaitForExit();
- Logger.Info(output);
- if (p.ExitCode != 0)
- {
- Logger.Info("Error: " + executable + " exited with code " + p.ExitCode + "\n");
- if (!string.IsNullOrEmpty(arguments))
- Logger.Info("Error: args: " + arguments + "\n");
- if (!string.IsNullOrEmpty(env))
- Logger.Info("Error: env: " + env + "\n");
- return false;
- }
- }
- catch (Exception e)
- {
- Logger.Info("Error: failed to launch " + executable + "\n");
- return false;
- }
-
- return true;
- }
-
- private static IEnumerable<ProjectItem> ParseProjectItems(object o)
- {
- ProjectItems subitems;
- if (o is Project)
- {
- subitems = (o as Project).ProjectItems;
- }
- else
- {
- yield return (o as ProjectItem);
- subitems = (o as ProjectItem).ProjectItems;
- }
-
- foreach (ProjectItem item in subitems)
- foreach (ProjectItem i in ParseProjectItems(item))
- yield return i;
- }
-
- private ServiceProvider sp;
-
- private List<Project> projects;
- }
-
- } /* namespace lol */
|