All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.cloudsimplus.allocationpolicies.VmAllocationPolicyAbstract Maven / Gradle / Ivy

/*
 * Title:        CloudSim Toolkit
 * Description:  CloudSim (Cloud Simulation) Toolkit for Modeling and Simulation of Clouds
 * Licence:      GPL - http://www.gnu.org/copyleft/gpl.html
 *
 * Copyright (c) 2009-2012, The University of Melbourne, Australia
 */
package org.cloudsimplus.allocationpolicies;

import lombok.Getter;
import lombok.NonNull;
import lombok.Setter;
import lombok.experimental.Accessors;
import org.cloudsimplus.autoscaling.VerticalVmScaling;
import org.cloudsimplus.datacenters.Datacenter;
import org.cloudsimplus.hosts.Host;
import org.cloudsimplus.hosts.HostSuitability;
import org.cloudsimplus.resources.Pe;
import org.cloudsimplus.resources.Processor;
import org.cloudsimplus.schedulers.MipsShare;
import org.cloudsimplus.util.Conversion;
import org.cloudsimplus.vms.Vm;
import org.cloudsimplus.vms.VmGroup;

import java.util.*;
import java.util.function.BiFunction;

import static java.util.stream.Collectors.toList;

/**
 * An abstract class that represents the policy
 * used by a {@link Datacenter} to choose a {@link Host} to place or migrate
 * a given {@link Vm}. It supports two-stage commit of reservation of
 * hosts: first, we reserve the Host and, once committed by the customer, the VM is
 * effectively allocated to that Host.
 *
 * 

Each {@link Datacenter} must to have its own instance of a {@link VmAllocationPolicy}.

* * @author Rodrigo N. Calheiros * @author Anton Beloglazov * @author Manoel Campos da Silva Filho * @since CloudSim Toolkit 1.0 */ @Accessors(makeFinal = false) @Getter @Setter public abstract class VmAllocationPolicyAbstract implements VmAllocationPolicy { /** * WARNING: the function should not be called directly because it may be null. * Use the {@link #findHostForVm(Vm)} instead. * * @see #setFindHostForVmFunction(BiFunction) */ private BiFunction> findHostForVmFunction; /** @see #getDatacenter() */ @NonNull private Datacenter datacenter; /** @see #getHostCountForParallelSearch() */ private int hostCountForParallelSearch; /** * Creates a VmAllocationPolicy. */ public VmAllocationPolicyAbstract() { //Passing null makes the class to use the default function to find a host for a VM. this(null); } /** * Creates a VmAllocationPolicy, changing the {@link BiFunction} to select a Host for a Vm. * * @param findHostForVmFunction a {@link BiFunction} to select a Host for a given Vm. * @see VmAllocationPolicy#setFindHostForVmFunction(BiFunction) */ public VmAllocationPolicyAbstract(final BiFunction> findHostForVmFunction) { setDatacenter(Datacenter.NULL); setFindHostForVmFunction(findHostForVmFunction); this.hostCountForParallelSearch = DEF_HOST_COUNT_PARALLEL_SEARCH; } @Override public final List getHostList() { return datacenter.getHostList(); } @Override public boolean scaleVmVertically(final VerticalVmScaling scaling) { if (scaling.isVmUnderloaded()) { return downScaleVmVertically(scaling); } if (scaling.isVmOverloaded()) { return upScaleVmVertically(scaling); } return false; } /** * Performs the up scaling of Vm resource associated to a given scaling object. * * @param scaling the Vm's scaling object * @return true if the Vm was overloaded and the up scaling was performed, false otherwise */ private boolean upScaleVmVertically(final VerticalVmScaling scaling) { return isRequestingCpuScaling(scaling) ? scaleVmPesUpOrDown(scaling) : upScaleVmNonCpuResource(scaling); } /** * Performs the down scaling of Vm resource associated to a given scaling object. * * @param scaling the Vm's scaling object * @return true if the down scaling was performed, false otherwise */ private boolean downScaleVmVertically(final VerticalVmScaling scaling) { return isRequestingCpuScaling(scaling) ? scaleVmPesUpOrDown(scaling) : downScaleVmNonCpuResource(scaling); } /** * Performs the up or down scaling of Vm {@link Pe}s, * depending if the VM is under or overloaded. * * @param scaling the Vm's scaling object * @return true if the scaling was performed, false otherwise * @see #upScaleVmVertically(VerticalVmScaling) */ private boolean scaleVmPesUpOrDown(final VerticalVmScaling scaling) { final double pesNumberForScaling = scaling.getResourceAmountToScale(); if (pesNumberForScaling == 0) { return false; } final boolean isVmUnderloaded = scaling.isVmUnderloaded(); //Avoids trying to downscale the number of vPEs to zero if(isVmUnderloaded && scaling.getVm().getPesNumber() == pesNumberForScaling) { scaling.logDownscaleToZeroNotAllowed(); return false; } if (scaling.isVmOverloaded() && isNotHostPesSuitableToUpScaleVm(scaling)) { scaling.logResourceUnavailable(); return false; } final Vm vm = scaling.getVm(); vm.getHost().getVmScheduler().deallocatePesFromVm(vm); final int signal = isVmUnderloaded ? -1 : 1; //Removes or adds some capacity from/to the resource, respectively if the VM is under or overloaded vm.getProcessor().sumCapacity((long) pesNumberForScaling * signal); vm.getHost().getVmScheduler().allocatePesForVm(vm); return true; } private boolean isNotHostPesSuitableToUpScaleVm(final VerticalVmScaling scaling) { final Vm vm = scaling.getVm(); final long pesCountForScaling = (long)scaling.getResourceAmountToScale(); final MipsShare additionalVmMips = new MipsShare(pesCountForScaling, vm.getMips()); return !vm.getHost().getVmScheduler().isSuitableForVm(vm, additionalVmMips); } /** * Checks if the scaling object is in charge of scaling CPU resource. * * @param scaling the Vm scaling object * @return true if the scaling is for CPU, false if it is * for any other kind of resource */ private boolean isRequestingCpuScaling(final VerticalVmScaling scaling) { return Processor.class.equals(scaling.getResourceClass()); } /** * Performs the up scaling of a Vm resource that is anything else than CPU. * * @param scaling the Vm's scaling object * @return true if the up scaling was performed, false otherwise * @see #scaleVmPesUpOrDown(VerticalVmScaling) * @see #upScaleVmVertically(VerticalVmScaling) */ private boolean upScaleVmNonCpuResource(final VerticalVmScaling scaling) { return scaling.allocateResourceForVm(); } /** * Performs the down scaling of a Vm resource that is anything else than CPU. * * @param scaling the Vm's scaling object * @return true if the down scaling was performed, false otherwise * @see #downScaleVmVertically(VerticalVmScaling) */ private boolean downScaleVmNonCpuResource(final VerticalVmScaling scaling) { final var resourceManageableClass = scaling.getResourceClass(); final var vmResource = scaling.getVm().getResource(resourceManageableClass); final double amountToDeallocate = scaling.getResourceAmountToScale(); final var resourceProvisioner = scaling.getVm().getHost().getProvisioner(resourceManageableClass); final double newTotalVmResource = vmResource.getCapacity() - amountToDeallocate; if (resourceProvisioner.allocateResourceForVm(scaling.getVm(), newTotalVmResource)) { LOGGER.info( "{}: {}: {} {} deallocated from {}: new capacity is {}. Current resource usage is {}%", scaling.getVm().getSimulation().clockStr(), scaling.getClass().getSimpleName(), (long) amountToDeallocate, resourceManageableClass.getSimpleName(), scaling.getVm(), vmResource.getCapacity(), vmResource.getPercentUtilization() * 100); return true; } LOGGER.error( "{}: {}: {} requested to reduce {} capacity by {} but an unexpected error occurred and the resource was not resized", scaling.getVm().getSimulation().clockStr(), scaling.getClass().getSimpleName(), scaling.getVm(), resourceManageableClass.getSimpleName(), (long) amountToDeallocate); return false; } /** * Allocates the host with less PEs in use for a given VM. * * @param vm {@inheritDoc} * @return {@inheritDoc} */ @Override public HostSuitability allocateHostForVm(final Vm vm) { if (getHostList().isEmpty()) { LOGGER.error( "{}: {}: {} could not be allocated because there isn't any Host for Datacenter {}", vm.getSimulation().clockStr(), getClass().getSimpleName(), vm, getDatacenter().getId()); return new HostSuitability("Datacenter has no host."); } if (vm.isCreated()) { return new HostSuitability("VM is already created"); } final var optionalHost = findHostForVm(vm); if (optionalHost.filter(Host::isActive).isPresent()) { return allocateHostForVm(vm, optionalHost.get()); } LOGGER.warn("{}: {}: No suitable host found for {} in {}", vm.getSimulation().clockStr(), getClass().getSimpleName(), vm, datacenter); return new HostSuitability("No suitable host found"); } @Override public List allocateHostForVm(@NonNull final Collection vmCollection) { return vmCollection.stream().filter(vm -> !allocateHostForVm(vm).fully()).collect(toList()); } @Override public HostSuitability allocateHostForVm(final Vm vm, final Host host) { if(vm instanceof VmGroup vmGroup){ return createVmsFromGroup(vmGroup, host); } return createVm(vm, host); } private HostSuitability createVmsFromGroup(final VmGroup vmGroup, final Host host) { int createdVms = 0; final var hostSuitabilityForVmGroup = new HostSuitability(); for (final Vm vm : vmGroup.getVmList()) { final var hostSuitability = createVm(vm, host); hostSuitabilityForVmGroup.setSuitability(hostSuitability); createdVms += Conversion.boolToInt(hostSuitability.fully()); } vmGroup.setCreated(createdVms > 0); if(vmGroup.isCreated()) { vmGroup.setHost(host); } return hostSuitabilityForVmGroup; } private HostSuitability createVm(final Vm vm, final Host host) { final var suitability = host.createVm(vm); if (suitability.fully()) { LOGGER.info( "{}: {}: {} has been allocated to {}", vm.getSimulation().clockStr(), getClass().getSimpleName(), vm, host); } else { LOGGER.error( "{}: {} Creation of {} on {} failed due to {}.", vm.getSimulation().clockStr(), getClass().getSimpleName(), vm, host, suitability); } return suitability; } @Override public void deallocateHostForVm(final Vm vm) { vm.getHost().destroyVm(vm); } /** * {@inheritDoc} * The default implementation of such a Function is provided by the method {@link #findHostForVm(Vm)}. * * @param findHostForVmFunction {@inheritDoc}. * Passing null makes the default method to find a Host for a VM to be used. */ @Override public final VmAllocationPolicy setFindHostForVmFunction(final BiFunction> findHostForVmFunction) { this.findHostForVmFunction = findHostForVmFunction; return this; } @Override public final Optional findHostForVm(final Vm vm) { final var optionalHost = findHostForVmFunction == null ? defaultFindHostForVm(vm) : findHostForVmFunction.apply(this, vm); //If the selected Host is not active, activate it (if it's already active, setActive has no effect) return optionalHost.map(host -> host.setActive(true)); } /** * Provides the default implementation of the policy * to find a suitable Host for a given VM. * * @param vm the VM to find a suitable Host to * @return an {@link Optional} containing a suitable Host to place the VM or an empty {@link Optional} if no suitable Host was found * @see #setFindHostForVmFunction(BiFunction) */ protected abstract Optional defaultFindHostForVm(Vm vm); @Override public Map getOptimizedAllocationMap(final List vmList) { /* * This method implementation doesn't perform any * VM placement optimization and, in fact, has no effect. * Classes implementing the {@link VmAllocationPolicyMigration} * provide actual implementations for this method that can be overridden * by subclasses. */ return Collections.emptyMap(); } @Override public boolean isVmMigrationSupported() { return false; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy