mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-26 06:25:19 +00:00
Do lazy init of host PM features
To avoid probing the host power management features on any call to virInitialize, only initialize the mutex in virNodeSuspendInit. Do lazy load of the supported PM target mask when it is actually needed * src/util/virnodesuspend.c: Lazy init of supported features
This commit is contained in:
parent
f1f28611f1
commit
2fc056c1ba
@ -46,9 +46,10 @@
|
|||||||
* Bitmask to hold the Power Management features supported by the host,
|
* Bitmask to hold the Power Management features supported by the host,
|
||||||
* such as Suspend-to-RAM, Suspend-to-Disk, Hybrid-Suspend etc.
|
* such as Suspend-to-RAM, Suspend-to-Disk, Hybrid-Suspend etc.
|
||||||
*/
|
*/
|
||||||
static unsigned int hostPMFeatures;
|
static unsigned int nodeSuspendTargetMask;
|
||||||
|
static bool nodeSuspendTargetMaskInit;
|
||||||
|
|
||||||
virMutex virNodeSuspendMutex;
|
static virMutex virNodeSuspendMutex;
|
||||||
|
|
||||||
static bool aboutToSuspend;
|
static bool aboutToSuspend;
|
||||||
|
|
||||||
@ -75,17 +76,9 @@ static void virNodeSuspendUnlock(void)
|
|||||||
*/
|
*/
|
||||||
int virNodeSuspendInit(void)
|
int virNodeSuspendInit(void)
|
||||||
{
|
{
|
||||||
|
|
||||||
if (virMutexInit(&virNodeSuspendMutex) < 0)
|
if (virMutexInit(&virNodeSuspendMutex) < 0)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
/* Get the power management capabilities supported by the host */
|
|
||||||
hostPMFeatures = 0;
|
|
||||||
if (virNodeSuspendGetTargetMask(&hostPMFeatures) < 0) {
|
|
||||||
if (geteuid() == 0)
|
|
||||||
VIR_ERROR(_("Failed to get host power management features"));
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -191,9 +184,14 @@ int nodeSuspendForDuration(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|||||||
{
|
{
|
||||||
static virThread thread;
|
static virThread thread;
|
||||||
char *cmdString = NULL;
|
char *cmdString = NULL;
|
||||||
|
int ret = -1;
|
||||||
|
unsigned int supported;
|
||||||
|
|
||||||
virCheckFlags(0, -1);
|
virCheckFlags(0, -1);
|
||||||
|
|
||||||
|
if (virNodeSuspendGetTargetMask(&supported) < 0)
|
||||||
|
return -1;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Ensure that we are the only ones trying to suspend.
|
* Ensure that we are the only ones trying to suspend.
|
||||||
* Fail if somebody has already initiated a suspend.
|
* Fail if somebody has already initiated a suspend.
|
||||||
@ -202,18 +200,16 @@ int nodeSuspendForDuration(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|||||||
|
|
||||||
if (aboutToSuspend) {
|
if (aboutToSuspend) {
|
||||||
/* A suspend operation is already in progress */
|
/* A suspend operation is already in progress */
|
||||||
virNodeSuspendUnlock();
|
virNodeSuspendError(VIR_ERR_OPERATION_INVALID, "%s",
|
||||||
return -1;
|
_("Suspend operation already in progress"));
|
||||||
} else {
|
goto cleanup;
|
||||||
aboutToSuspend = true;
|
|
||||||
}
|
}
|
||||||
|
aboutToSuspend = true;
|
||||||
virNodeSuspendUnlock();
|
|
||||||
|
|
||||||
/* Check if the host supports the requested suspend target */
|
/* Check if the host supports the requested suspend target */
|
||||||
switch (target) {
|
switch (target) {
|
||||||
case VIR_NODE_SUSPEND_TARGET_MEM:
|
case VIR_NODE_SUSPEND_TARGET_MEM:
|
||||||
if (hostPMFeatures & (1 << VIR_NODE_SUSPEND_TARGET_MEM)) {
|
if (supported & (1 << VIR_NODE_SUSPEND_TARGET_MEM)) {
|
||||||
cmdString = strdup("pm-suspend");
|
cmdString = strdup("pm-suspend");
|
||||||
if (cmdString == NULL) {
|
if (cmdString == NULL) {
|
||||||
virReportOOMError();
|
virReportOOMError();
|
||||||
@ -225,7 +221,7 @@ int nodeSuspendForDuration(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
case VIR_NODE_SUSPEND_TARGET_DISK:
|
case VIR_NODE_SUSPEND_TARGET_DISK:
|
||||||
if (hostPMFeatures & (1 << VIR_NODE_SUSPEND_TARGET_DISK)) {
|
if (supported & (1 << VIR_NODE_SUSPEND_TARGET_DISK)) {
|
||||||
cmdString = strdup("pm-hibernate");
|
cmdString = strdup("pm-hibernate");
|
||||||
if (cmdString == NULL) {
|
if (cmdString == NULL) {
|
||||||
virReportOOMError();
|
virReportOOMError();
|
||||||
@ -237,7 +233,7 @@ int nodeSuspendForDuration(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
case VIR_NODE_SUSPEND_TARGET_HYBRID:
|
case VIR_NODE_SUSPEND_TARGET_HYBRID:
|
||||||
if (hostPMFeatures & (1 << VIR_NODE_SUSPEND_TARGET_HYBRID)) {
|
if (supported & (1 << VIR_NODE_SUSPEND_TARGET_HYBRID)) {
|
||||||
cmdString = strdup("pm-suspend-hybrid");
|
cmdString = strdup("pm-suspend-hybrid");
|
||||||
if (cmdString == NULL) {
|
if (cmdString == NULL) {
|
||||||
virReportOOMError();
|
virReportOOMError();
|
||||||
@ -263,11 +259,11 @@ int nodeSuspendForDuration(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
ret = 0;
|
||||||
|
|
||||||
cleanup:
|
cleanup:
|
||||||
|
virNodeSuspendUnlock();
|
||||||
VIR_FREE(cmdString);
|
VIR_FREE(cmdString);
|
||||||
return -1;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -338,35 +334,40 @@ cleanup:
|
|||||||
int
|
int
|
||||||
virNodeSuspendGetTargetMask(unsigned int *bitmask)
|
virNodeSuspendGetTargetMask(unsigned int *bitmask)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret = -1;
|
||||||
bool supported;
|
|
||||||
|
|
||||||
*bitmask = 0;
|
*bitmask = 0;
|
||||||
|
|
||||||
|
virNodeSuspendLock();
|
||||||
|
/* Get the power management capabilities supported by the host */
|
||||||
|
if (!nodeSuspendTargetMaskInit) {
|
||||||
|
bool supported;
|
||||||
|
nodeSuspendTargetMask = 0;
|
||||||
|
|
||||||
/* Check support for Suspend-to-RAM (S3) */
|
/* Check support for Suspend-to-RAM (S3) */
|
||||||
ret = virNodeSuspendSupportsTarget(VIR_NODE_SUSPEND_TARGET_MEM, &supported);
|
if (virNodeSuspendSupportsTarget(VIR_NODE_SUSPEND_TARGET_MEM, &supported) < 0)
|
||||||
if (ret < 0)
|
goto cleanup;
|
||||||
goto error;
|
|
||||||
if (supported)
|
if (supported)
|
||||||
*bitmask |= (1 << VIR_NODE_SUSPEND_TARGET_MEM);
|
nodeSuspendTargetMask |= (1 << VIR_NODE_SUSPEND_TARGET_MEM);
|
||||||
|
|
||||||
/* Check support for Suspend-to-Disk (S4) */
|
/* Check support for Suspend-to-Disk (S4) */
|
||||||
ret = virNodeSuspendSupportsTarget(VIR_NODE_SUSPEND_TARGET_DISK, &supported);
|
if (virNodeSuspendSupportsTarget(VIR_NODE_SUSPEND_TARGET_DISK, &supported) < 0)
|
||||||
if (ret < 0)
|
goto cleanup;
|
||||||
goto error;
|
|
||||||
if (supported)
|
if (supported)
|
||||||
*bitmask |= (1 << VIR_NODE_SUSPEND_TARGET_DISK);
|
nodeSuspendTargetMask |= (1 << VIR_NODE_SUSPEND_TARGET_DISK);
|
||||||
|
|
||||||
/* Check support for Hybrid-Suspend */
|
/* Check support for Hybrid-Suspend */
|
||||||
ret = virNodeSuspendSupportsTarget(VIR_NODE_SUSPEND_TARGET_HYBRID, &supported);
|
if (virNodeSuspendSupportsTarget(VIR_NODE_SUSPEND_TARGET_HYBRID, &supported) < 0)
|
||||||
if (ret < 0)
|
goto cleanup;
|
||||||
goto error;
|
|
||||||
if (supported)
|
if (supported)
|
||||||
*bitmask |= (1 << VIR_NODE_SUSPEND_TARGET_HYBRID);
|
nodeSuspendTargetMask |= (1 << VIR_NODE_SUSPEND_TARGET_HYBRID);
|
||||||
|
|
||||||
return 0;
|
nodeSuspendTargetMaskInit = true;
|
||||||
|
}
|
||||||
error:
|
|
||||||
*bitmask = 0;
|
*bitmask = nodeSuspendTargetMask;
|
||||||
return -1;
|
ret = 0;
|
||||||
|
cleanup:
|
||||||
|
virNodeSuspendUnlock();
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user