/* * Copyright (c) 2020 Alibaba Group Holding Limited. All Rights Reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Alibaba designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. */ package com.alibaba.tenant; import java.lang.ref.WeakReference; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.LinkedList; import java.util.Map; import java.util.Properties; import java.util.WeakHashMap; import java.util.concurrent.atomic.AtomicLong; import java.util.function.Supplier; import java.util.concurrent.locks.ReentrantReadWriteLock; import com.alibaba.rcm.ResourceContainer; import com.alibaba.rcm.Constraint; import com.alibaba.rcm.internal.AbstractResourceContainer; import static com.alibaba.tenant.TenantState.*; /** * TenantContainer is a "virtual container" for a tenant of application, the * resource consumption of tenant such as CPU, heap is constrained by the policy * of this "virtual container". The thread can run in virtual container by * calling TenantContainer.run * */ public class TenantContainer { TenantResourceContainer resourceContainer; private static NativeDispatcher nd = new NativeDispatcher(); /* * Used to generate the tenant id. */ private static AtomicLong nextTenantID = new AtomicLong(0); /* * Used to hold the mapping from tenant id to TenantContainer object for all * tenants */ private static Map tenantContainerMap = null; /* * Holds the threads attached with this tenant container */ private List attachedThreads = new LinkedList<>(); /* * Newly created threads which attach to this tenant container */ private List> spawnedThreads = Collections.synchronizedList(new ArrayList<>()); /* * Used to contain service threads, including finalizer threads, shutdown hook threads. */ private Map serviceThreads = Collections.synchronizedMap(new WeakHashMap<>()); /* * the configuration of this tenant container */ private TenantConfiguration configuration = null; /* * tenant state */ private volatile TenantState state; /* * tenant id */ private long tenantId; /* * address of native tenant allocation context */ private long allocationContext = 0L; /* * tenant name */ private String name; /* * Used to store the system properties per tenant */ private Properties props; /* * allocated memory of attached threads, which is accumulated for current tenant */ private long accumulatedMemory = 0L; /** * Get total allocated memory of this tenant. * @return the total allocated memory of this tenant. */ public synchronized long getAllocatedMemory() { Thread[] threads = getAttachedThreads(); int size = threads.length; long[] ids = new long[size]; long[] memSizes = new long[size]; for (int i = 0; i < size; i++) { ids[i] = threads[i].getId(); } nd.getThreadsAllocatedMemory(ids, memSizes); long totalThreadsAllocatedMemory = 0; for (long s : memSizes) { totalThreadsAllocatedMemory += s; } return totalThreadsAllocatedMemory + accumulatedMemory; } /** * Used to track and run tenant shutdown hooks */ private TenantShutdownHooks tenantShutdownHooks = new TenantShutdownHooks(); /** * Sets the tenant properties to the one specified by argument. * @param props the properties to be set, CoW the system properties if it is null. */ public void setProperties(Properties props) { if (props == null) { props = new Properties(); Properties sysProps = System.getProperties(); for(Object key: sysProps.keySet()) { props.put(key, sysProps.get(key)); } } this.props = props; } /** * Gets the properties of tenant * @return the tenant properties */ public Properties getProperties() { return props; } /** * Sets the property indicated by the specified key. * @param key the name of the property. * @param value the value of the property. * @return the previous value of the property, * or null if it did not have one. */ public String setProperty(String key, String value) { checkKey(key); return (String) props.setProperty(key, value); } /** * Gets the property indicated by the specified key. * @param key the name of the property. * @return the string value of the property, * or null if there is no property with that key. */ public String getProperty(String key) { checkKey(key); return props.getProperty(key); } /** * Removes the property indicated by the specified key. * @param key the name of the property to be removed. * @return the previous string value of the property, * or null if there was no property with that key. */ public String clearProperty(String key) { checkKey(key); return (String) props.remove(key); } private void checkKey(String key) { if (null == key) { throw new NullPointerException("key can't be null"); } if ("".equals(key)) { throw new IllegalArgumentException("key can't be empty"); } } // // Used to synchronize between destroy() and runThread() private ReentrantReadWriteLock destroyLock = new ReentrantReadWriteLock(); /** * Destroy this tenant container and release occupied resources including memory, cpu, FD, etc. * */ public void destroy() { if (TenantContainer.current() != null) { throw new RuntimeException("Should only call destroy() in ROOT tenant"); } destroyLock.writeLock().lock(); try { if (state != TenantState.STOPPING && state != TenantState.DEAD) { setState(TenantState.STOPPING); tenantContainerMap.remove(getTenantId()); // finish all finalizers resourceContainer.attach(); nd.attach(this); try { Runtime.getRuntime().runFinalization(); } finally { nd.attach(null); resourceContainer.detach(); } // execute all shutdown hooks tenantShutdownHooks.runHooks(); } } catch (Throwable t) { System.err.println("Exception from TenantContainer.destroy()"); t.printStackTrace(); } finally { setState(TenantState.DEAD); cleanUp(); destroyLock.writeLock().unlock(); } } /* * Release all native resources and Java references * should be the very last step of {@link #destroy()} operation. * If cannot kill all threads in {@link #killAllThreads()}, should do this in {@link WatchDogThread} * */ private void cleanUp() { if (TenantGlobals.isHeapIsolationEnabled()) { nd.destroyTenantAllocationContext(allocationContext); } resourceContainer.destroyImpl(); // clear references spawnedThreads.clear(); attachedThreads.clear(); tenantShutdownHooks = null; } private TenantContainer(TenantContainer parent, String name, TenantConfiguration configuration) { this.tenantId = nextTenantID.getAndIncrement(); this.resourceContainer = new TenantResourceContainer( (parent == null ? null : parent.resourceContainer), this, configuration.getAllConstraints()); this.name = (name == null ? "Tenant-" + getTenantId() : name); this.configuration = configuration; setState(STARTING); //Initialize the tenant properties. props = new Properties(); props.putAll(System.getProperties()); tenantContainerMap.put(this.tenantId, this); // Create allocation context if heap isolation enabled if (TenantGlobals.isHeapIsolationEnabled()) { nd.createTenantAllocationContext(this, configuration.getMaxHeap()); } } TenantConfiguration getConfiguration() { return configuration; } /** * @return the tenant state */ public TenantState getState() { return state; } /* * Set the tenant state * @param state used to set */ void setState(TenantState state) { this.state = state; } /** * Returns the tenant' id * @return tenant id */ public long getTenantId() { return tenantId; } /** * Returns this tenant's name. * @return this tenant's name. */ public String getName() { return name; } /** * @return A collection of all threads attached to the container. */ public synchronized Thread[] getAttachedThreads() { return attachedThreads.toArray(new Thread[attachedThreads.size()]); } /** * Get the tenant container by id * @param id tenant id. * @return the tenant specified by id, null if the id doesn't exist. */ public static TenantContainer getTenantContainerById(long id) { checkIfTenantIsEnabled(); return tenantContainerMap.get(id); } /** * Create tenant container by the configuration * @param configuration used to create tenant * @return the tenant container */ public static TenantContainer create(TenantConfiguration configuration) { return create(TenantContainer.current(), configuration); } /** * Create tenant container by the configuration * @param parent parent tenant container * @param configuration used to create tenant * @return the tenant container */ public static TenantContainer create(TenantContainer parent, TenantConfiguration configuration) { checkIfTenantIsEnabled(); //parameter checking if (null == configuration) { throw new IllegalArgumentException("Failed to create tenant, illegal arguments: configuration is null"); } return create(parent, null, configuration); } /** * Create tenant container by the name and configuration * @param name the tenant name * @param configuration used to create tenant * @return the tenant container */ public static TenantContainer create(String name, TenantConfiguration configuration) { return create(TenantContainer.current(), name, configuration); } /** * Create tenant container by the name and configuration * @param parent parent tenant container * @param name the tenant name * @param configuration used to create tenant * @return the tenant container */ public static TenantContainer create(TenantContainer parent, String name, TenantConfiguration configuration) { checkIfTenantIsEnabled(); //parameter checking if (null == configuration) { throw new IllegalArgumentException("Failed to create tenant, illegal arguments: configuration is null"); } TenantContainer tc = new TenantContainer(parent, name, configuration); tenantContainerMap.put(tc.getTenantId(), tc); return tc; } /** * Gets the tenant id list * @return the tenant id list, Collections.emptyList if no tenant exists. */ public static List getAllTenantIds() { checkIfTenantIsEnabled(); if (null == tenantContainerMap) { throw new IllegalStateException("TenantContainer class is not initialized !"); } if (tenantContainerMap.size() == 0) { return Collections.EMPTY_LIST; } return new ArrayList<>(tenantContainerMap.keySet()); } /** * Gets the TenantContainer attached to the current thread. * @return The TenantContainer attached to the current thread, null if no * TenantContainer is attached to the current thread. */ public static TenantContainer current() { checkIfTenantIsEnabled(); AbstractResourceContainer curResContainer = TenantResourceContainer.current(); if (TenantResourceContainer.root() == curResContainer) { return null; } assert curResContainer instanceof TenantResourceContainer; return ((TenantResourceContainer)curResContainer).getTenant(); } /** * Gets the cpu time consumed by this tenant * @return the cpu time used by this tenant, 0 if tenant cpu throttling or accounting feature is disabled. */ public long getProcessCpuTime() { if (!TenantGlobals.isCpuAccountingEnabled()) { throw new IllegalStateException("-XX:+TenantCpuAccounting is not enabled"); } long cpuTime = 0; return cpuTime; } /** * Gets the heap space occupied by this tenant * @return heap space occupied by this tenant, 0 if tenant heap isolation is disabled. * @throws IllegalStateException if -XX:+TenantHeapIsolation is not enabled. */ public long getOccupiedMemory() { if (!TenantGlobals.isHeapIsolationEnabled()) { throw new IllegalStateException("-XX:+TenantHeapIsolation is not enabled"); } return nd.getTenantOccupiedMemory(allocationContext); } /** * Runs the code in the target tenant container * @param task the code to run */ public void run(Runnable task) throws TenantException { if (getState() == DEAD || getState() == STOPPING) { throw new TenantException("Tenant is dead"); } TenantContainer container = current(); assert container != null; if (container == this) { task.run(); } else { if (container != null) { throw new TenantException("must be in root container " + "before running into non-root container."); } attach(); try { task.run(); } finally { detach(); } } } /* * Get accumulatedMemory value of current thread */ private long getThreadAllocatedMemory() { long[] memSizes = new long[1]; nd.getThreadsAllocatedMemory(null, memSizes); return memSizes[0]; } private void attach() { // This is the first thread which runs in this tenant container if (getState() == TenantState.STARTING) { // move the tenant state to RUNNING this.setState(TenantState.RUNNING); } Thread curThread = Thread.currentThread(); long curAllocBytes = getThreadAllocatedMemory(); synchronized (this) { attachedThreads.add(curThread); accumulatedMemory -= curAllocBytes; resourceContainer.attach(); nd.attach(this); } } private void detach() { Thread curThread = Thread.currentThread(); long curAllocBytes = getThreadAllocatedMemory(); synchronized (this) { nd.attach(null); resourceContainer.detach(); attachedThreads.remove(curThread); accumulatedMemory += curAllocBytes; } } /* * Check if the tenant feature is enabled. */ private static void checkIfTenantIsEnabled() { if (!TenantGlobals.isTenantEnabled()) { throw new UnsupportedOperationException("The multi-tenant feature is not enabled!"); } } /* * Invoked by the VM to run a thread in multi-tenant mode. * * NOTE: please ensure relevant logic has been fully understood before changing any code * * @throws TenantException */ private void runThread(final Thread thread) throws TenantException { if (destroyLock.readLock().tryLock()) { if (getState() != STOPPING && getState() != DEAD) { spawnedThreads.add(new WeakReference<>(thread)); this.run(() -> { destroyLock.readLock().unlock(); thread.run(); }); } else { destroyLock.readLock().unlock(); } // try to clean up once if (destroyLock.readLock().tryLock()) { if (getState() != STOPPING && getState() != DEAD) { spawnedThreads.removeIf(ref -> ref.get() == null || ref.get() == thread); } destroyLock.readLock().unlock(); } } else { // shutdown in progress if (serviceThreads.containsKey(thread)) { // attach to current thread to run without registering resourceContainer.attach(); nd.attach(this); try { thread.run(); } finally { nd.attach(null); resourceContainer.detach(); removeServiceThread(thread); } } } } /* * Initialize the TenantContainer class, called after System.initializeSystemClass by VM. */ private static void initializeTenantContainerClass() { //Initialize this field after the system is booted. tenantContainerMap = Collections.synchronizedMap(new HashMap()); try { // force initialization of TenantConfiguration Class.forName("com.alibaba.tenant.TenantConfiguration"); } catch (ClassNotFoundException e) { e.printStackTrace(); } } /** * Retrieve the tenant container where obj is allocated in * @param obj object to be searched * @return TenantContainer object whose memory space contains obj, * or null if ROOT tenant container */ public static TenantContainer containerOf(Object obj) { if (!TenantGlobals.isHeapIsolationEnabled()) { throw new UnsupportedOperationException("containerOf() only works with -XX:+TenantHeapIsolation"); } return obj != null ? nd.containerOf(obj) : null; } /** * Runs {@code Supplier.get} in the root tenant. * @param supplier target used to call * @return the result of {@code Supplier.get} */ public static T primitiveRunInRoot(Supplier supplier) { // thread is already in root tenant. if(null == TenantContainer.current()) { return supplier.get(); } else{ TenantContainer tenant = TenantContainer.current(); //Force to root tenant. tenant.resourceContainer.detach(); nd.attach(null); try { T t = supplier.get(); return t; } finally { nd.attach(tenant); tenant.resourceContainer.attach(); } } } /** * Runs a block of code in the root tenant. * @param runnable the code to run */ public static void primitiveRunInRoot(Runnable runnable) { primitiveRunInRoot(() -> { runnable.run(); return null; }); } /** * Register a new tenant shutdown hook. * When the tenant begins its destroy it will * start all registered shutdown hooks in some unspecified order and let * them run concurrently. * @param hook * An initialized but unstarted {@link Thread} object */ public void addShutdownHook(Thread hook) { addServiceThread(hook); tenantShutdownHooks.add(hook); } /** * De-registers a previously-registered tenant shutdown hook. * @param hook the hook to remove * @return true if the specified hook had previously been * registered and was successfully de-registered, false * otherwise. */ public boolean removeShutdownHook(Thread hook) { removeServiceThread(hook); return tenantShutdownHooks.remove(hook); } // add a thread to the service thread list private void addServiceThread(Thread thread) { if (thread != null) { serviceThreads.put(thread, null); } } // remove a thread from the service thread list private void removeServiceThread(Thread thread) { serviceThreads.remove(thread); } /** * Try to modify resource limit of current tenant, * for resource whose limit cannot be changed after creation of {@code TenantContainer}, its limit will be ignored. * @param config new TenantConfiguration to */ public void update(TenantConfiguration config) { for (Constraint constraint : config.getAllConstraints()) { updateConstraint(constraint); } } void updateConstraint(Constraint constraint) { resourceContainer.updateConstraint(constraint); getConfiguration().setConstraint(constraint); } /** * @return {@code ResourceContainer} of this tenant */ public ResourceContainer getResourceContainer() { return resourceContainer; } }