462 lines
13 KiB
462 lines
13 KiB
/* Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
file Copyright.txt or https://cmake.org/licensing for details. */
|
|
#include "cmCallVisualStudioMacro.h"
|
|
|
|
#include <sstream>
|
|
|
|
#include "cmStringAlgorithms.h"
|
|
#include "cmSystemTools.h"
|
|
|
|
#if defined(_MSC_VER)
|
|
# define HAVE_COMDEF_H
|
|
#endif
|
|
|
|
// Just for this file:
|
|
//
|
|
static bool LogErrorsAsMessages;
|
|
|
|
#if defined(HAVE_COMDEF_H)
|
|
|
|
# include <comdef.h>
|
|
|
|
// Copied from a correct comdef.h to avoid problems with deficient versions
|
|
// of comdef.h that exist in the wild... Fixes issue #7533.
|
|
//
|
|
# ifdef _NATIVE_WCHAR_T_DEFINED
|
|
# ifdef _DEBUG
|
|
# pragma comment(lib, "comsuppwd.lib")
|
|
# else
|
|
# pragma comment(lib, "comsuppw.lib")
|
|
# endif
|
|
# else
|
|
# ifdef _DEBUG
|
|
# pragma comment(lib, "comsuppd.lib")
|
|
# else
|
|
# pragma comment(lib, "comsupp.lib")
|
|
# endif
|
|
# endif
|
|
|
|
//! Use ReportHRESULT to make a cmSystemTools::Message after calling
|
|
//! a COM method that may have failed.
|
|
# define ReportHRESULT(hr, context) \
|
|
if (FAILED(hr)) { \
|
|
if (LogErrorsAsMessages) { \
|
|
std::ostringstream _hresult_oss; \
|
|
_hresult_oss.flags(std::ios::hex); \
|
|
_hresult_oss << context << " failed HRESULT, hr = 0x" << hr << '\n'; \
|
|
_hresult_oss.flags(std::ios::dec); \
|
|
_hresult_oss << __FILE__ << "(" << __LINE__ << ")"; \
|
|
cmSystemTools::Message(_hresult_oss.str()); \
|
|
} \
|
|
}
|
|
|
|
//! Using the given instance of Visual Studio, call the named macro
|
|
HRESULT InstanceCallMacro(IDispatch* vsIDE, const std::string& macro,
|
|
const std::string& args)
|
|
{
|
|
HRESULT hr = E_POINTER;
|
|
|
|
_bstr_t macroName(macro.c_str());
|
|
_bstr_t macroArgs(args.c_str());
|
|
|
|
if (0 != vsIDE) {
|
|
DISPID dispid = (DISPID)-1;
|
|
wchar_t execute_command[] = L"ExecuteCommand";
|
|
OLECHAR* name = execute_command;
|
|
|
|
hr =
|
|
vsIDE->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_USER_DEFAULT, &dispid);
|
|
ReportHRESULT(hr, "GetIDsOfNames(ExecuteCommand)");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
VARIANTARG vargs[2];
|
|
DISPPARAMS params;
|
|
VARIANT result;
|
|
EXCEPINFO excep;
|
|
UINT arg = (UINT)-1;
|
|
|
|
// No VariantInit or VariantClear calls are necessary for
|
|
// these two vargs. They are both local _bstr_t variables
|
|
// that remain in scope for the duration of the Invoke call.
|
|
//
|
|
V_VT(&vargs[1]) = VT_BSTR;
|
|
V_BSTR(&vargs[1]) = macroName;
|
|
V_VT(&vargs[0]) = VT_BSTR;
|
|
V_BSTR(&vargs[0]) = macroArgs;
|
|
|
|
params.rgvarg = &vargs[0];
|
|
params.rgdispidNamedArgs = 0;
|
|
params.cArgs = sizeof(vargs) / sizeof(vargs[0]);
|
|
params.cNamedArgs = 0;
|
|
|
|
VariantInit(&result);
|
|
|
|
memset(&excep, 0, sizeof(excep));
|
|
|
|
hr = vsIDE->Invoke(dispid, IID_NULL, LOCALE_USER_DEFAULT,
|
|
DISPATCH_METHOD, ¶ms, &result, &excep, &arg);
|
|
|
|
std::ostringstream oss;
|
|
/* clang-format off */
|
|
oss << "\nInvoke(ExecuteCommand)\n"
|
|
" Macro: " << macro << "\n"
|
|
" Args: " << args << '\n';
|
|
/* clang-format on */
|
|
|
|
if (DISP_E_EXCEPTION == hr) {
|
|
/* clang-format off */
|
|
oss << "DISP_E_EXCEPTION EXCEPINFO:" << excep.wCode << "\n"
|
|
" wCode: " << excep.wCode << "\n"
|
|
" wReserved: " << excep.wReserved << '\n';
|
|
/* clang-format on */
|
|
if (excep.bstrSource) {
|
|
oss << " bstrSource: " << (const char*)(_bstr_t)excep.bstrSource
|
|
<< '\n';
|
|
}
|
|
if (excep.bstrDescription) {
|
|
oss << " bstrDescription: "
|
|
<< (const char*)(_bstr_t)excep.bstrDescription << '\n';
|
|
}
|
|
if (excep.bstrHelpFile) {
|
|
oss << " bstrHelpFile: " << (const char*)(_bstr_t)excep.bstrHelpFile
|
|
<< '\n';
|
|
}
|
|
/* clang-format off */
|
|
oss << " dwHelpContext: " << excep.dwHelpContext << "\n"
|
|
" pvReserved: " << excep.pvReserved << "\n"
|
|
" pfnDeferredFillIn: "
|
|
<< reinterpret_cast<void*>(excep.pfnDeferredFillIn) << "\n"
|
|
" scode: " << excep.scode << '\n';
|
|
/* clang-format on */
|
|
}
|
|
|
|
std::string exstr(oss.str());
|
|
ReportHRESULT(hr, exstr.c_str());
|
|
|
|
VariantClear(&result);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//! Get the Solution object from the IDE object
|
|
HRESULT GetSolutionObject(IDispatch* vsIDE, IDispatchPtr& vsSolution)
|
|
{
|
|
HRESULT hr = E_POINTER;
|
|
|
|
if (0 != vsIDE) {
|
|
DISPID dispid = (DISPID)-1;
|
|
wchar_t solution[] = L"Solution";
|
|
OLECHAR* name = solution;
|
|
|
|
hr =
|
|
vsIDE->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_USER_DEFAULT, &dispid);
|
|
ReportHRESULT(hr, "GetIDsOfNames(Solution)");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
DISPPARAMS params;
|
|
VARIANT result;
|
|
EXCEPINFO excep;
|
|
UINT arg = (UINT)-1;
|
|
|
|
params.rgvarg = 0;
|
|
params.rgdispidNamedArgs = 0;
|
|
params.cArgs = 0;
|
|
params.cNamedArgs = 0;
|
|
|
|
VariantInit(&result);
|
|
|
|
memset(&excep, 0, sizeof(excep));
|
|
|
|
hr = vsIDE->Invoke(dispid, IID_NULL, LOCALE_USER_DEFAULT,
|
|
DISPATCH_PROPERTYGET, ¶ms, &result, &excep, &arg);
|
|
ReportHRESULT(hr, "Invoke(Solution)");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
vsSolution = V_DISPATCH(&result);
|
|
}
|
|
|
|
VariantClear(&result);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//! Get the FullName property from the Solution object
|
|
HRESULT GetSolutionFullName(IDispatch* vsSolution, std::string& fullName)
|
|
{
|
|
HRESULT hr = E_POINTER;
|
|
|
|
if (0 != vsSolution) {
|
|
DISPID dispid = (DISPID)-1;
|
|
wchar_t full_name[] = L"FullName";
|
|
OLECHAR* name = full_name;
|
|
|
|
hr = vsSolution->GetIDsOfNames(IID_NULL, &name, 1, LOCALE_USER_DEFAULT,
|
|
&dispid);
|
|
ReportHRESULT(hr, "GetIDsOfNames(FullName)");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
DISPPARAMS params;
|
|
VARIANT result;
|
|
EXCEPINFO excep;
|
|
UINT arg = (UINT)-1;
|
|
|
|
params.rgvarg = 0;
|
|
params.rgdispidNamedArgs = 0;
|
|
params.cArgs = 0;
|
|
params.cNamedArgs = 0;
|
|
|
|
VariantInit(&result);
|
|
|
|
memset(&excep, 0, sizeof(excep));
|
|
|
|
hr = vsSolution->Invoke(dispid, IID_NULL, LOCALE_USER_DEFAULT,
|
|
DISPATCH_PROPERTYGET, ¶ms, &result, &excep,
|
|
&arg);
|
|
ReportHRESULT(hr, "Invoke(FullName)");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
fullName = (std::string)(_bstr_t)V_BSTR(&result);
|
|
}
|
|
|
|
VariantClear(&result);
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//! Get the FullName property from the Solution object, given the IDE object
|
|
HRESULT GetIDESolutionFullName(IDispatch* vsIDE, std::string& fullName)
|
|
{
|
|
IDispatchPtr vsSolution;
|
|
HRESULT hr = GetSolutionObject(vsIDE, vsSolution);
|
|
ReportHRESULT(hr, "GetSolutionObject");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
GetSolutionFullName(vsSolution, fullName);
|
|
ReportHRESULT(hr, "GetSolutionFullName");
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//! Get all running objects from the Windows running object table.
|
|
//! Save them in a map by their display names.
|
|
HRESULT GetRunningInstances(std::map<std::string, IUnknownPtr>& mrot)
|
|
{
|
|
// mrot == Map of the Running Object Table
|
|
|
|
IRunningObjectTablePtr runningObjectTable;
|
|
IEnumMonikerPtr monikerEnumerator;
|
|
IMonikerPtr moniker;
|
|
ULONG numFetched = 0;
|
|
|
|
HRESULT hr = GetRunningObjectTable(0, &runningObjectTable);
|
|
ReportHRESULT(hr, "GetRunningObjectTable");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
hr = runningObjectTable->EnumRunning(&monikerEnumerator);
|
|
ReportHRESULT(hr, "EnumRunning");
|
|
}
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
hr = monikerEnumerator->Reset();
|
|
ReportHRESULT(hr, "Reset");
|
|
}
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
while (S_OK == monikerEnumerator->Next(1, &moniker, &numFetched)) {
|
|
std::string runningObjectName;
|
|
IUnknownPtr runningObjectVal;
|
|
IBindCtxPtr ctx;
|
|
|
|
hr = CreateBindCtx(0, &ctx);
|
|
ReportHRESULT(hr, "CreateBindCtx");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
LPOLESTR displayName = 0;
|
|
hr = moniker->GetDisplayName(ctx, 0, &displayName);
|
|
ReportHRESULT(hr, "GetDisplayName");
|
|
if (displayName) {
|
|
runningObjectName = (std::string)(_bstr_t)displayName;
|
|
CoTaskMemFree(displayName);
|
|
}
|
|
|
|
hr = runningObjectTable->GetObject(moniker, &runningObjectVal);
|
|
ReportHRESULT(hr, "GetObject");
|
|
if (SUCCEEDED(hr)) {
|
|
mrot.insert(std::make_pair(runningObjectName, runningObjectVal));
|
|
}
|
|
}
|
|
|
|
numFetched = 0;
|
|
moniker = 0;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
//! Do the two file names refer to the same Visual Studio solution? Or are
|
|
//! we perhaps looking for any and all solutions?
|
|
bool FilesSameSolution(const std::string& slnFile, const std::string& slnName)
|
|
{
|
|
if (slnFile == "ALL" || slnName == "ALL") {
|
|
return true;
|
|
}
|
|
|
|
// Otherwise, make lowercase local copies, convert to Unix slashes, and
|
|
// see if the resulting strings are the same:
|
|
std::string s1 = cmSystemTools::LowerCase(slnFile);
|
|
std::string s2 = cmSystemTools::LowerCase(slnName);
|
|
cmSystemTools::ConvertToUnixSlashes(s1);
|
|
cmSystemTools::ConvertToUnixSlashes(s2);
|
|
|
|
return s1 == s2;
|
|
}
|
|
|
|
//! Find instances of Visual Studio with the given solution file
|
|
//! open. Pass "ALL" for slnFile to gather all running instances
|
|
//! of Visual Studio.
|
|
HRESULT FindVisualStudioInstances(const std::string& slnFile,
|
|
std::vector<IDispatchPtr>& instances)
|
|
{
|
|
std::map<std::string, IUnknownPtr> mrot;
|
|
|
|
HRESULT hr = GetRunningInstances(mrot);
|
|
ReportHRESULT(hr, "GetRunningInstances");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
std::map<std::string, IUnknownPtr>::iterator it;
|
|
for (it = mrot.begin(); it != mrot.end(); ++it) {
|
|
if (cmHasLiteralPrefix(it->first, "!VisualStudio.DTE.")) {
|
|
IDispatchPtr disp(it->second);
|
|
if (disp != (IDispatch*)0) {
|
|
std::string slnName;
|
|
hr = GetIDESolutionFullName(disp, slnName);
|
|
ReportHRESULT(hr, "GetIDESolutionFullName");
|
|
|
|
if (FilesSameSolution(slnFile, slnName)) {
|
|
instances.push_back(disp);
|
|
|
|
// std::cout << "Found Visual Studio instance." << std::endl;
|
|
// std::cout << " ROT entry name: " << it->first << std::endl;
|
|
// std::cout << " ROT entry object: "
|
|
// << (IUnknown*) it->second << std::endl;
|
|
// std::cout << " slnFile: " << slnFile << std::endl;
|
|
// std::cout << " slnName: " << slnName << std::endl;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
#endif // defined(HAVE_COMDEF_H)
|
|
|
|
int cmCallVisualStudioMacro::GetNumberOfRunningVisualStudioInstances(
|
|
const std::string& slnFile)
|
|
{
|
|
int count = 0;
|
|
|
|
LogErrorsAsMessages = false;
|
|
|
|
#if defined(HAVE_COMDEF_H)
|
|
HRESULT hr = CoInitialize(0);
|
|
ReportHRESULT(hr, "CoInitialize");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
std::vector<IDispatchPtr> instances;
|
|
hr = FindVisualStudioInstances(slnFile, instances);
|
|
ReportHRESULT(hr, "FindVisualStudioInstances");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
count = static_cast<int>(instances.size());
|
|
}
|
|
|
|
// Force release all COM pointers before CoUninitialize:
|
|
instances.clear();
|
|
|
|
CoUninitialize();
|
|
}
|
|
#else
|
|
(void)slnFile;
|
|
#endif
|
|
|
|
return count;
|
|
}
|
|
|
|
//! Get all running objects from the Windows running object table.
|
|
//! Save them in a map by their display names.
|
|
int cmCallVisualStudioMacro::CallMacro(const std::string& slnFile,
|
|
const std::string& macro,
|
|
const std::string& args,
|
|
const bool logErrorsAsMessages)
|
|
{
|
|
int err = 1; // no comdef.h
|
|
|
|
LogErrorsAsMessages = logErrorsAsMessages;
|
|
|
|
#if defined(HAVE_COMDEF_H)
|
|
err = 2; // error initializing
|
|
|
|
HRESULT hr = CoInitialize(0);
|
|
ReportHRESULT(hr, "CoInitialize");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
std::vector<IDispatchPtr> instances;
|
|
hr = FindVisualStudioInstances(slnFile, instances);
|
|
ReportHRESULT(hr, "FindVisualStudioInstances");
|
|
|
|
if (SUCCEEDED(hr)) {
|
|
err = 0; // no error
|
|
|
|
std::vector<IDispatchPtr>::iterator it;
|
|
for (it = instances.begin(); it != instances.end(); ++it) {
|
|
hr = InstanceCallMacro(*it, macro, args);
|
|
ReportHRESULT(hr, "InstanceCallMacro");
|
|
|
|
if (FAILED(hr)) {
|
|
err = 3; // error attempting to call the macro
|
|
}
|
|
}
|
|
|
|
if (instances.empty()) {
|
|
// no instances to call
|
|
|
|
// cmSystemTools::Message(
|
|
// "cmCallVisualStudioMacro::CallMacro no instances found to call",
|
|
// "Warning");
|
|
}
|
|
}
|
|
|
|
// Force release all COM pointers before CoUninitialize:
|
|
instances.clear();
|
|
|
|
CoUninitialize();
|
|
}
|
|
#else
|
|
(void)slnFile;
|
|
(void)macro;
|
|
(void)args;
|
|
if (LogErrorsAsMessages) {
|
|
cmSystemTools::Message("cmCallVisualStudioMacro::CallMacro is not "
|
|
"supported on this platform");
|
|
}
|
|
#endif
|
|
|
|
if (err && LogErrorsAsMessages) {
|
|
std::ostringstream oss;
|
|
oss << "cmCallVisualStudioMacro::CallMacro failed, err = " << err;
|
|
cmSystemTools::Message(oss.str());
|
|
}
|
|
|
|
return 0;
|
|
}
|