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

org.apache.ratis.util.ResourceSemaphore Maven / Gradle / Ivy

There is a newer version: 3.1.2
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.apache.ratis.util;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.concurrent.Semaphore;
import java.util.concurrent.atomic.AtomicBoolean;

/**
 * A {@link Semaphore} with a limit for a resource.
 *
 * After {@link #close()}, the resource becomes unavailable, i.e. any acquire will not succeed.
 */
public class ResourceSemaphore extends Semaphore {
  private final int limit;
  private final AtomicBoolean reducePermits = new AtomicBoolean();
  private final AtomicBoolean isClosed = new AtomicBoolean();

  public ResourceSemaphore(int limit) {
    super(limit, true);
    Preconditions.assertTrue(limit > 0, () -> "limit = " + limit + " <= 0");
    this.limit = limit;
  }

  @Override
  public void release() {
    release(1);
  }

  @Override
  public void release(int permits) {
    assertRelease(permits);
    super.release(permits);
    assertAvailable();
  }

  private void assertRelease(int toRelease) {
    Preconditions.assertTrue(toRelease >= 0, () -> "toRelease = " + toRelease + " < 0");
    final int available = assertAvailable();
    final int permits = Math.addExact(available, toRelease);
    Preconditions.assertTrue(permits <= limit, () -> "permits = " + permits + " > limit = " + limit);
  }

  private int assertAvailable() {
    final int available = availablePermits();
    Preconditions.assertTrue(available >= 0, () -> "available = " + available + " < 0");
    return available;
  }

  public int used() {
    return limit - availablePermits();
  }

  /** Close the resource. */
  public void close() {
    if (reducePermits.compareAndSet(false, true)) {
      reducePermits(limit);
      isClosed.set(true);
    }
  }

  public boolean isClosed() {
    return isClosed.get();
  }

  @Override
  public String toString() {
    return (isClosed()? "closed/": availablePermits() + "/") + limit;
  }

  /**
   * Track a group of resources with a list of {@link ResourceSemaphore}s.
   */
  public static class Group {
    public static final int SUCCESS = -1;

    private final List resources;

    public Group(int... limits) {
      Preconditions.assertTrue(limits.length >= 1, () -> "limits is empty");
      final List list = new ArrayList<>(limits.length);
      for(int limit : limits) {
        list.add(new ResourceSemaphore(limit));
      }
      this.resources = Collections.unmodifiableList(list);
    }

    public int resourceSize() {
      return resources.size();
    }

    protected ResourceSemaphore get(int i) {
      return resources.get(i);
    }

    /** @return {@link #SUCCESS} if successfully acquired; otherwise, return the failed index. */
    public int tryAcquire(int... permits) {
      Preconditions.assertTrue(permits.length == resources.size(),
          () -> "items.length = " + permits.length + " != resources.size() = " + resources.size());
      int i = 0;
      // try acquiring all resources
      for(; i < permits.length; i++) {
        if (!resources.get(i).tryAcquire(permits[i])) {
          break;
        }
      }

      if (i == permits.length) {
        return SUCCESS; // successfully acquired all resources
      }

      // failed at i, releasing all previous resources
      for(int k = i - 1; k >= 0; k--) {
        resources.get(k).release(permits[k]);
      }

      return i;
    }

    public void acquire(int... permits) throws InterruptedException {
      Preconditions.assertTrue(permits.length == resources.size(),
          () -> "items.length = " + permits.length + " != resources.size() = "
              + resources.size());
      int i = 0;
      try {
        for (; i < permits.length; i++) {
          resources.get(i).acquire(permits[i]);
        }
      } catch (Exception e) {
        for (; --i >= 0;) {
          resources.get(i).release(permits[i]);
        }
        throw e;
      }
    }

    protected void release(int... permits) {
      for(int i = resources.size() - 1; i >= 0; i--) {
        resources.get(i).release(permits[i]);
      }
    }

    public void close() {
      for(int i = resources.size() - 1; i >= 0; i--) {
        resources.get(i).close();
      }
    }

    public boolean isClosed() {
      return resources.get(resources.size() - 1).isClosed();
    }

    @Override
    public String toString() {
      return resources + ",size=" + resources.size();
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy