Changeset 75663 in vbox
- Timestamp:
- Nov 22, 2018 2:00:59 PM (6 years ago)
- svn:sync-xref-src-repo-rev:
- 126855
- Location:
- trunk/src/VBox/Main
- Files:
-
- 7 edited
Legend:
- Unmodified
- Added
- Removed
-
trunk/src/VBox/Main/idl/VirtualBox.xidl
r75648 r75663 25520 25520 <interface 25521 25521 name="ICloudProvider" extends="$unknown" 25522 uuid=" 8154fa84-35d7-4a9b-86b4-cb2635b0d9f5"25522 uuid="caaf8bc1-aab0-4ea0-b825-be53a1453bdd" 25523 25523 wsmap="managed" reservedMethods="4" reservedAttributes="4" 25524 25524 > … … 25604 25604 <param name="profileName" type="wstring" dir="in"/> 25605 25605 <param name="profile" type="ICloudProfile" dir="return"/> 25606 </method> 25607 25608 <method name="prepareUninstall" wrap-hint-server="passcaller"> 25609 <desc> 25610 The caller requests the cloud provider to cease operation. Should 25611 return an error if this is currently not possible (due to ongoing 25612 cloud activity, possibly by a different API client). However, this 25613 must not wait for the completion for a larger amount of time (ideally 25614 stays below a second of execution time). If this succeeds it should 25615 leave the cloud provider in a state which does not allow starting new 25616 operations. 25617 </desc> 25606 25618 </method> 25607 25619 -
trunk/src/VBox/Main/include/CloudProviderManagerImpl.h
r74926 r75663 21 21 22 22 #include "CloudProviderManagerWrap.h" 23 #ifdef VBOX_WITH_EXTPACK24 class ExtPackManager;25 #endif26 23 27 24 … … 35 32 void FinalRelease(); 36 33 37 HRESULT init( VirtualBox *aParent);34 HRESULT init(); 38 35 void uninit(); 39 36 40 37 #ifdef VBOX_WITH_EXTPACK 41 38 // Safe helpers, take care of caller and lock themselves. 42 void i_refreshProviders(); 39 bool i_canRemoveExtPack(IExtPack *aExtPack); 40 void i_addExtPack(IExtPack *aExtPack); 43 41 #endif 44 42 … … 55 53 private: 56 54 #ifdef VBOX_WITH_EXTPACK 57 ComObjPtr<ExtPackManager> mpExtPackMgr; 58 uint64_t mcExtPackMgrUpdate; 59 std::map<com::Utf8Str, ComPtr<ICloudProviderManager> > m_mapCloudProviderManagers; 55 typedef std::map<com::Utf8Str, ComPtr<ICloudProviderManager> > ExtPackNameCloudProviderManagerMap; 56 ExtPackNameCloudProviderManagerMap m_mapCloudProviderManagers; 57 typedef std::vector<com::Utf8Str> ExtPackNameVec; 58 ExtPackNameVec m_astrExtPackNames; 60 59 #endif 61 60 62 std::vector<ComPtr<ICloudProvider> > m_apCloudProviders; 61 typedef std::vector<ComPtr<ICloudProvider> > CloudProviderVec; 62 CloudProviderVec m_apCloudProviders; 63 63 }; 64 64 -
trunk/src/VBox/Main/include/ExtPackManagerImpl.h
r74804 r75663 131 131 bool i_wantsToBeDefaultVrde(void) const; 132 132 HRESULT i_refresh(bool *pfCanDelete); 133 #ifndef VBOX_COM_INPROC 134 bool i_areThereCloudProviderUninstallVetos(); 135 void i_notifyCloudProviderManager(); 136 #endif 133 137 /** @} */ 134 138 -
trunk/src/VBox/Main/include/VirtualBoxImpl.h
r75380 r75663 40 40 class DHCPServer; 41 41 class PerformanceCollector; 42 class CloudProviderManager; 42 43 #ifdef VBOX_WITH_EXTPACK 43 44 class ExtPackManager; … … 216 217 const Guid &i_getGlobalRegistryId() const; 217 218 218 const ComObjPtr<Host>& i_host() const; 219 SystemProperties* i_getSystemProperties() const; 219 const ComObjPtr<Host> &i_host() const; 220 SystemProperties *i_getSystemProperties() const; 221 CloudProviderManager *i_getCloudProviderManager() const; 220 222 #ifdef VBOX_WITH_EXTPACK 221 ExtPackManager *i_getExtPackManager() const;223 ExtPackManager *i_getExtPackManager() const; 222 224 #endif 223 225 #ifdef VBOX_WITH_RESOURCE_USAGE_API 224 const ComObjPtr<PerformanceCollector> &i_performanceCollector() const;226 const ComObjPtr<PerformanceCollector> &i_performanceCollector() const; 225 227 #endif /* VBOX_WITH_RESOURCE_USAGE_API */ 226 228 … … 229 231 230 232 /** Returns the VirtualBox home directory */ 231 const Utf8Str &i_homeDir() const;233 const Utf8Str &i_homeDir() const; 232 234 int i_calculateFullPath(const Utf8Str &strPath, Utf8Str &aResult); 233 235 void i_copyPathRelativeToConfig(const Utf8Str &strSource, Utf8Str &strTarget); -
trunk/src/VBox/Main/src-all/ExtPackManagerImpl.cpp
r74814 r75663 21 21 *********************************************************************************************************************************/ 22 22 #include "ExtPackManagerImpl.h" 23 #include "CloudProviderManagerImpl.h" 23 24 #include "ExtPackUtil.h" 24 25 #include "ThreadTask.h" … … 928 929 a_pLock->release(); 929 930 m->pReg->pfnVirtualBoxReady(m->pReg, a_pVirtualBox); 931 i_notifyCloudProviderManager(); 930 932 a_pLock->acquire(); 931 933 return true; … … 1085 1087 return false; 1086 1088 } 1087 #endif /* !VBOX_COM_INPROC */1089 #endif /* VBOX_COM_INPROC */ 1088 1090 1089 1091 /** … … 1248 1250 return S_OK; 1249 1251 } 1252 1253 #ifndef VBOX_COM_INPROC 1254 /** 1255 * Checks if there are cloud providers vetoing extension pack uninstall. 1256 * 1257 * This needs going through the cloud provider singleton in VirtualBox, 1258 * the job cannot be done purely by using the code in the extension pack). 1259 * It cannot be integrated into i_callUninstallHookAndClose, because it 1260 * can only do its job when the extpack lock is not held, whereas the 1261 * actual uninstall must be done with the lock held all the time for 1262 * consistency reasons. 1263 * 1264 * This is called when uninstalling or replacing an extension pack. 1265 * 1266 * @returns true / false 1267 */ 1268 bool ExtPack::i_areThereCloudProviderUninstallVetos() 1269 { 1270 Assert(m->pVirtualBox != NULL); /* Only called from VBoxSVC. */ 1271 1272 ComObjPtr<CloudProviderManager> cpm(m->pVirtualBox->i_getCloudProviderManager()); 1273 AssertReturn(!cpm.isNull(), false); 1274 1275 return !cpm->i_canRemoveExtPack(static_cast<IExtPack *>(this)); 1276 } 1277 1278 /** 1279 * Notifies the Cloud Provider Manager that there is a new extension pack. 1280 * 1281 * This is called when installing an extension pack. 1282 * 1283 * @param a_pExtPack The extension pack to be added. 1284 */ 1285 void ExtPack::i_notifyCloudProviderManager() 1286 { 1287 Assert(m->pVirtualBox != NULL); /* Only called from VBoxSVC. */ 1288 1289 ComObjPtr<CloudProviderManager> cpm(m->pVirtualBox->i_getCloudProviderManager()); 1290 AssertReturnVoid(!cpm.isNull()); 1291 1292 cpm->i_addExtPack(static_cast<IExtPack *>(this)); 1293 } 1294 1295 #endif /* !VBOX_COM_INPROC */ 1250 1296 1251 1297 /** … … 2849 2895 autoLock.release(); 2850 2896 bool fRunningVMs = i_areThereAnyRunningVMs(); 2897 bool fVetoingCP = pExtPack->i_areThereCloudProviderUninstallVetos(); 2851 2898 autoLock.acquire(); 2852 2899 hrc = i_refreshExtPack(pStrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack); 2853 2900 if (fRunningVMs) 2854 2901 { 2855 LogRel(("Install extension pack '%s' failed because at least one VM is still running.", pStrName->c_str())); 2856 hrc = setError(E_FAIL, tr("Install extension pack '%s' failed because at least one VM is still running"), 2902 LogRel(("Upgrading extension pack '%s' failed because at least one VM is still running.", pStrName->c_str())); 2903 hrc = setError(E_FAIL, tr("Upgrading extension pack '%s' failed because at least one VM is still running"), 2904 pStrName->c_str()); 2905 } 2906 else if (fVetoingCP) 2907 { 2908 LogRel(("Upgrading extension pack '%s' failed because at least one Cloud Provider is still busy.", pStrName->c_str())); 2909 hrc = setError(E_FAIL, tr("Upgrading extension pack '%s' failed because at least one Cloud Provider is still busy"), 2857 2910 pStrName->c_str()); 2858 2911 } … … 2956 3009 if (SUCCEEDED(hrc)) 2957 3010 { 2958 if (a_fForcedRemoval || !i_areThereAnyRunningVMs()) 2959 { 2960 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS); 2961 2962 /* 2963 * Refresh the data we have on the extension pack as it may be made 2964 * stale by direct meddling or some other user. 2965 */ 2966 ExtPack *pExtPack; 2967 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack); 2968 if (SUCCEEDED(hrc)) 3011 AutoWriteLock autoLock(this COMMA_LOCKVAL_SRC_POS); 3012 3013 /* 3014 * Refresh the data we have on the extension pack as it 3015 * may be made stale by direct meddling or some other user. 3016 */ 3017 ExtPack *pExtPack; 3018 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack); 3019 if (SUCCEEDED(hrc) && pExtPack) 3020 { 3021 /* We must leave the lock when calling i_areThereAnyRunningVMs, 3022 which means we have to redo the refresh call afterwards. */ 3023 autoLock.release(); 3024 bool fRunningVMs = i_areThereAnyRunningVMs(); 3025 bool fVetoingCP = pExtPack->i_areThereCloudProviderUninstallVetos(); 3026 autoLock.acquire(); 3027 if (a_fForcedRemoval || (!fRunningVMs && !fVetoingCP)) 2969 3028 { 2970 if (!pExtPack) 3029 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack); 3030 if (SUCCEEDED(hrc)) 2971 3031 { 2972 LogRel(("ExtPackManager: Extension pack '%s' is not installed, so nothing to uninstall.\n", a_pstrName->c_str())); 2973 hrc = S_OK; /* nothing to uninstall */ 2974 } 2975 else 2976 { 2977 /* 2978 * Call the uninstall hook and unload the main dll. 2979 */ 2980 hrc = pExtPack->i_callUninstallHookAndClose(m->pVirtualBox, a_fForcedRemoval); 2981 if (SUCCEEDED(hrc)) 3032 if (!pExtPack) 3033 { 3034 LogRel(("ExtPackManager: Extension pack '%s' is not installed, so nothing to uninstall.\n", a_pstrName->c_str())); 3035 hrc = S_OK; /* nothing to uninstall */ 3036 } 3037 else 2982 3038 { 2983 3039 /* 2984 * Run the set-uid-to-root binary that performs the 2985 * uninstallation. Then refresh the object. 2986 * 2987 * This refresh is theorically subject to races, but it's of 2988 * the don't-do-that variety. 3040 * Call the uninstall hook and unload the main dll. 2989 3041 */ 2990 const char *pszForcedOpt = a_fForcedRemoval ? "--forced" : NULL; 2991 hrc = i_runSetUidToRootHelper(a_pstrDisplayInfo, 2992 "uninstall", 2993 "--base-dir", m->strBaseDir.c_str(), 2994 "--name", a_pstrName->c_str(), 2995 pszForcedOpt, /* Last as it may be NULL. */ 2996 (const char *)NULL); 3042 hrc = pExtPack->i_callUninstallHookAndClose(m->pVirtualBox, a_fForcedRemoval); 2997 3043 if (SUCCEEDED(hrc)) 2998 3044 { 2999 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack); 3045 /* 3046 * Run the set-uid-to-root binary that performs the 3047 * uninstallation. Then refresh the object. 3048 * 3049 * This refresh is theorically subject to races, but it's of 3050 * the don't-do-that variety. 3051 */ 3052 const char *pszForcedOpt = a_fForcedRemoval ? "--forced" : NULL; 3053 hrc = i_runSetUidToRootHelper(a_pstrDisplayInfo, 3054 "uninstall", 3055 "--base-dir", m->strBaseDir.c_str(), 3056 "--name", a_pstrName->c_str(), 3057 pszForcedOpt, /* Last as it may be NULL. */ 3058 (const char *)NULL); 3000 3059 if (SUCCEEDED(hrc)) 3001 3060 { 3002 if (!pExtPack) 3003 LogRel(("ExtPackManager: Successfully uninstalled extension pack '%s'.\n", a_pstrName->c_str())); 3004 else 3005 hrc = setError(E_FAIL, 3006 tr("Uninstall extension pack '%s' failed under mysterious circumstances"), 3007 a_pstrName->c_str()); 3061 hrc = i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, &pExtPack); 3062 if (SUCCEEDED(hrc)) 3063 { 3064 if (!pExtPack) 3065 LogRel(("ExtPackManager: Successfully uninstalled extension pack '%s'.\n", a_pstrName->c_str())); 3066 else 3067 hrc = setError(E_FAIL, 3068 tr("Uninstall extension pack '%s' failed under mysterious circumstances"), 3069 a_pstrName->c_str()); 3070 } 3008 3071 } 3009 }3010 else3011 {3012 ErrorInfoKeeper Eik;3013 i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, NULL);3072 else 3073 { 3074 ErrorInfoKeeper Eik; 3075 i_refreshExtPack(a_pstrName->c_str(), false /*a_fUnusableIsError*/, NULL); 3076 } 3014 3077 } 3015 3078 } 3016 3079 } 3017 3080 } 3018 } 3019 else 3020 { 3021 LogRel(("Uninstall extension pack '%s' failed because at least one VM is still running.", a_pstrName->c_str())); 3022 hrc = setError(E_FAIL, tr("Uninstall extension pack '%s' failed because at least one VM is still running"), 3023 a_pstrName->c_str()); 3081 else 3082 { 3083 if (fRunningVMs) 3084 { 3085 LogRel(("Uninstall extension pack '%s' failed because at least one VM is still running.", a_pstrName->c_str())); 3086 hrc = setError(E_FAIL, tr("Uninstall extension pack '%s' failed because at least one VM is still running"), 3087 a_pstrName->c_str()); 3088 } 3089 else if (fVetoingCP) 3090 { 3091 LogRel(("Uninstall extension pack '%s' failed because at least one Cloud Provider is still busy.", a_pstrName->c_str())); 3092 hrc = setError(E_FAIL, tr("Uninstall extension pack '%s' failed because at least one Cloud Provider is still busy"), 3093 a_pstrName->c_str()); 3094 } 3095 else 3096 { 3097 LogRel(("Uninstall extension pack '%s' failed for an unknown reason.", a_pstrName->c_str())); 3098 hrc = setError(E_FAIL, tr("Uninstall extension pack '%s' failed for an unknown reason"), 3099 a_pstrName->c_str()); 3100 3101 } 3102 } 3024 3103 } 3025 3104 … … 3029 3108 */ 3030 3109 if (m->enmContext == VBOXEXTPACKCTX_PER_USER_DAEMON) 3110 { 3111 autoLock.release(); 3031 3112 i_callAllVirtualBoxReadyHooks(); 3113 } 3032 3114 } 3033 3115 -
trunk/src/VBox/Main/src-server/CloudProviderManagerImpl.cpp
r74926 r75663 25 25 #include "Logging.h" 26 26 27 using namespace std;28 27 29 28 //////////////////////////////////////////////////////////////////////////////// … … 53 52 } 54 53 55 HRESULT CloudProviderManager::init( VirtualBox *aParent)54 HRESULT CloudProviderManager::init() 56 55 { 57 56 // Enclose the state transition NotReady->InInit->Ready. … … 61 60 m_apCloudProviders.clear(); 62 61 63 #ifdef VBOX_WITH_EXTPACK64 // Engage the extension pack manager and get all the implementations of65 // this class and all implemented cloud providers.66 ExtPackManager *pExtPackMgr = aParent->i_getExtPackManager();67 mpExtPackMgr = pExtPackMgr;68 if (pExtPackMgr)69 {70 mcExtPackMgrUpdate = pExtPackMgr->i_getUpdateCounter();71 // Make sure that the current value doesn't match, forcing a refresh.72 mcExtPackMgrUpdate--;73 i_refreshProviders();74 }75 #else76 RT_NOREF(aParent);77 #endif78 79 62 autoInitSpan.setSucceeded(); 80 63 return S_OK; … … 90 73 91 74 #ifdef VBOX_WITH_EXTPACK 92 void CloudProviderManager::i_refreshProviders() 93 { 94 uint64_t cExtPackMgrUpdate; 95 { 96 AutoReadLock alock(this COMMA_LOCKVAL_SRC_POS); 97 if (mpExtPackMgr.isNull()) 98 return; 99 cExtPackMgrUpdate = mpExtPackMgr->i_getUpdateCounter(); 100 if (cExtPackMgrUpdate == mcExtPackMgrUpdate) 101 return; 102 } 75 bool CloudProviderManager::i_canRemoveExtPack(IExtPack *aExtPack) 76 { 77 AssertReturn(aExtPack, false); 103 78 104 79 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); 105 // Reread the current value to figure out if some other thead did the work 106 // already before this thread got hold of the write lock. 107 cExtPackMgrUpdate = mpExtPackMgr->i_getUpdateCounter(); 108 if (cExtPackMgrUpdate == mcExtPackMgrUpdate) 109 return; 110 mcExtPackMgrUpdate = cExtPackMgrUpdate; 111 112 if (!m_mapCloudProviderManagers.empty()) 113 { 114 /// @todo The code below will need to be extended to handle extpack 115 // install and uninstall safely (and the latter needs non-trivial 116 // checks if any extpack related cloud activity is pending. 117 return; 118 } 119 120 std::vector<ComPtr<IUnknown> > apObjects; 80 81 // If any cloud provider in this extension pack fails to prepare the 82 // uninstall it and the cloud provider will be kept, so that the user 83 // can retry safely later. All other cloud providers in this extpack 84 // will be done as usual. No attempt is made to bring back the other 85 // cloud providers into working shape. 86 87 bool fRes = true; 88 Bstr bstrName; 89 aExtPack->GetName(bstrName.asOutParam()); 90 Utf8Str strName(bstrName); 91 ExtPackNameCloudProviderManagerMap::iterator it = m_mapCloudProviderManagers.find(strName); 92 if (it != m_mapCloudProviderManagers.end()) 93 { 94 ComPtr<ICloudProviderManager> pTmp(it->second); 95 96 Assert(m_astrExtPackNames.size() == m_apCloudProviders.size()); 97 for (size_t i = 0; i < m_astrExtPackNames.size(); ) 98 { 99 if (m_astrExtPackNames[i] != strName) 100 { 101 i++; 102 continue; 103 } 104 105 // pTmpProvider will point to an object with refcount > 0 until 106 // the ComPtr is removed from m_apCloudProviders. 107 HRESULT hrc = S_OK; 108 ULONG uRefCnt = 1; 109 ICloudProvider *pTmpProvider(m_apCloudProviders[i]); 110 if (pTmpProvider) 111 { 112 hrc = pTmpProvider->PrepareUninstall(); 113 // Sanity check the refcount, it should be 1 at this point. 114 pTmpProvider->AddRef(); 115 uRefCnt = pTmpProvider->Release(); 116 Assert(uRefCnt == 1); 117 } 118 if (SUCCEEDED(hrc) && uRefCnt == 1) 119 { 120 m_astrExtPackNames.erase(m_astrExtPackNames.begin() + i); 121 m_apCloudProviders.erase(m_apCloudProviders.begin() + i); 122 } 123 else 124 { 125 LogRel(("CloudProviderManager: provider '%s' blocks extpack uninstall, result=%Rhrc, refcount=%u\n", strName.c_str(), hrc, uRefCnt)); 126 fRes = false; 127 i++; 128 } 129 } 130 131 if (fRes) 132 m_mapCloudProviderManagers.erase(it); 133 } 134 135 return fRes; 136 } 137 138 void CloudProviderManager::i_addExtPack(IExtPack *aExtPack) 139 { 140 AssertReturnVoid(aExtPack); 141 142 AutoWriteLock alock(this COMMA_LOCKVAL_SRC_POS); 143 144 Bstr bstrName; 145 aExtPack->GetName(bstrName.asOutParam()); 146 Utf8Str strName(bstrName); 147 ComPtr<IUnknown> pObj; 121 148 std::vector<com::Utf8Str> astrExtPackNames; 122 149 com::Guid idObj(COM_IIDOF(ICloudProviderManager)); 123 mpExtPackMgr->i_queryObjects(idObj.toString(), apObjects, &astrExtPackNames); 124 for (unsigned i = 0; i < apObjects.size(); i++) 125 { 126 ComPtr<ICloudProviderManager> pTmp; 127 HRESULT hrc = apObjects[i].queryInterfaceTo(pTmp.asOutParam()); 128 if (SUCCEEDED(hrc)) 129 m_mapCloudProviderManagers[astrExtPackNames[i]] = pTmp; 130 SafeIfaceArray<ICloudProvider> apProvidersFromCurrExtPack; 131 hrc = pTmp->COMGETTER(Providers)(ComSafeArrayAsOutParam(apProvidersFromCurrExtPack)); 132 if (SUCCEEDED(hrc)) 133 { 134 for (unsigned j = 0; j < apProvidersFromCurrExtPack.size(); j++) 135 m_apCloudProviders.push_back(apProvidersFromCurrExtPack[i]); 136 } 150 HRESULT hrc = aExtPack->QueryObject(Bstr(idObj.toString()).raw(), pObj.asOutParam()); 151 if (FAILED(hrc)) 152 return; 153 154 ComPtr<ICloudProviderManager> pTmp(pObj); 155 if (pTmp.isNull()) 156 return; 157 158 SafeIfaceArray<ICloudProvider> apProvidersFromCurrExtPack; 159 hrc = pTmp->COMGETTER(Providers)(ComSafeArrayAsOutParam(apProvidersFromCurrExtPack)); 160 if (FAILED(hrc)) 161 return; 162 163 m_mapCloudProviderManagers[strName] = pTmp; 164 for (unsigned i = 0; i < apProvidersFromCurrExtPack.size(); i++) 165 { 166 Assert(m_astrExtPackNames.size() == m_apCloudProviders.size()); 167 m_astrExtPackNames.push_back(strName); 168 m_apCloudProviders.push_back(apProvidersFromCurrExtPack[i]); 137 169 } 138 170 } -
trunk/src/VBox/Main/src-server/VirtualBoxImpl.cpp
r75380 r75663 555 555 rc = unconst(m->pCloudProviderManager).createObject(); 556 556 if (SUCCEEDED(rc)) 557 rc = m->pCloudProviderManager->init( this);557 rc = m->pCloudProviderManager->init(); 558 558 ComAssertComRCThrowRC(rc); 559 559 if (FAILED(rc)) throw rc; … … 3768 3768 } 3769 3769 3770 CloudProviderManager *VirtualBox::i_getCloudProviderManager() const 3771 { 3772 return m->pCloudProviderManager; 3773 } 3774 3770 3775 #ifdef VBOX_WITH_EXTPACK 3771 3776 /**
Note:
See TracChangeset
for help on using the changeset viewer.