blockmanager.java

来自「RESIN 3.2 最新源码」· Java 代码 · 共 380 行

JAVA
380
字号
/* * Copyright (c) 1998-2008 Caucho Technology -- all rights reserved * * This file is part of Resin(R) Open Source * * Each copy or derived work must preserve the copyright notice and this * notice unmodified. * * Resin Open Source is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * Resin Open Source 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, or any warranty * of NON-INFRINGEMENT.  See the GNU General Public License for more * details. * * You should have received a copy of the GNU General Public License * along with Resin Open Source; if not, write to the * *   Free Software Foundation, Inc. *   59 Temple Place, Suite 330 *   Boston, MA 02111-1307  USA * * @author Scott Ferguson */package com.caucho.db.store;import com.caucho.log.Log;import com.caucho.management.server.AbstractManagedObject;import com.caucho.management.server.BlockManagerMXBean;import com.caucho.util.L10N;import com.caucho.util.LongKeyLruCache;import java.io.IOException;import java.util.ArrayList;import java.util.Iterator;import java.util.logging.Level;import java.util.logging.Logger;/** * Manages the block cache */public final class BlockManager  extends AbstractManagedObject  implements BlockManagerMXBean{  private static final Logger log = Log.open(BlockManager.class);  private static final L10N L = new L10N(BlockManager.class);  private static BlockManager _staticManager;  private final byte []_storeMask = new byte[8192];  private final LongKeyLruCache<Block> _blockCache;  private final ArrayList<Block> _writeQueue = new ArrayList<Block>();  private int _writeQueueMax = 32;  private BlockManager(int capacity)  {    _blockCache = new LongKeyLruCache<Block>(capacity);    // the first store id is not available to allow for tests for zero.    _storeMask[0] |= 1;    registerSelf();    BlockManagerWriter writer = new BlockManagerWriter();    Thread thread = new Thread(writer, "resin-block-manager-writer");    thread.setDaemon(true);    thread.start();  }  /**   * Returns the block manager, ensuring a minimum number of entries.   */  public static synchronized BlockManager create(int minEntries)  {    if (_staticManager == null)      _staticManager = new BlockManager(minEntries);    _staticManager.ensureCapacity(minEntries);    return _staticManager;  }  public static BlockManager getBlockManager()  {    return _staticManager;  }  /**   * Ensures the cache has a minimum number of blocks.   *   * @param minCapacity the minimum capacity in blocks   */  public void ensureCapacity(int minCapacity)  {    _blockCache.ensureCapacity(minCapacity);  }  /**   * Allocates a store id.   */  public int allocateStoreId()  {    synchronized (_storeMask) {      for (int i = 0; i < _storeMask.length; i++) {	int mask = _storeMask[i];	if (mask != 0xff) {	  for (int j = 0; j < 8; j++) {	    if ((mask & (1 << j)) == 0) {	      _storeMask[i] |= (1 << j);	      return 8 * i + j;	    }	  }	}      }      throw new IllegalStateException(L.l("All store ids used."));    }  }  /**   * Frees blocks with the given store.   */  public void flush(Store store)  {    ArrayList<Block> dirtyBlocks = null;    synchronized (_blockCache) {      Iterator<Block> values = _blockCache.values();      while (values.hasNext()) {	Block block = values.next();	if (block != null && block.getStore() == store) {	  if (block.isDirty()) {	    if (dirtyBlocks == null)	      dirtyBlocks = new ArrayList<Block>();	    dirtyBlocks.add(block);	  }	}      }    }    for (int i = 0; dirtyBlocks != null && i < dirtyBlocks.size(); i++) {      Block block = dirtyBlocks.get(i);      try {	synchronized (block) {	  block.write();	}      } catch (IOException e) {	log.log(Level.FINER, e.toString(), e);      }    }  }  /**   * Frees blocks with the given store.   */  public void freeStore(Store store)  {    ArrayList<Block> removeBlocks = new ArrayList<Block>();    synchronized (_blockCache) {      Iterator<Block> iter = _blockCache.values();      while (iter.hasNext()) {	Block block = iter.next();	if (block != null && block.getStore() == store)	  removeBlocks.add(block);      }    }    for (Block block : removeBlocks) {      _blockCache.remove(block.getBlockId());    }    synchronized (_writeQueue) {      while (_writeQueue.size() > 0) {	try {	  _writeQueue.wait();	} catch (InterruptedException e) {	}      }    }  }  /**   * Frees a store id.   */  public void freeStoreId(int storeId)  {    synchronized (_storeMask) {      if (storeId <= 0)	throw new IllegalArgumentException(String.valueOf(storeId));      _storeMask[storeId / 8] &= ~(1 << storeId % 8);    }  }  /**   * Gets the table's block.   */  Block getBlock(Store store, long blockId)  {    // XXX: proper handling of the synchronized is tricky because    // the LRU dirty write might have timing issues        Block block = _blockCache.get(blockId);    while (block == null || ! block.allocate()) {      // Find any matching block in the process of being written      Block dirtyBlock = null;      synchronized (_writeQueue) {	int size = _writeQueue.size();		for (int i = 0; i < size; i++) {	  dirtyBlock = _writeQueue.get(i);	  if (dirtyBlock.getBlockId() == blockId && dirtyBlock.allocate()) {	    break;	  }	  else	    dirtyBlock = null;	}      }          if ((blockId & Store.BLOCK_MASK) == 0)	throw stateError(L.l("Block 0 is reserved."));      block = new ReadBlock(store, blockId);      if (dirtyBlock != null) {	byte []dirtyBuffer = dirtyBlock.getBuffer();	if (dirtyBuffer != null) {	  System.arraycopy(dirtyBuffer, 0, block.getBuffer(), 0, dirtyBuffer.length);	  block.validate();	}	dirtyBlock.free();      }      // needs to be outside the synchronized since the put      // can cause an LRU drop which might lead to a dirty write      block = _blockCache.putIfNew(blockId, block);    }    return block;  }  /**   * Adds a block that's needs to be flushed.   */  void addLruDirtyWriteBlock(Block block)  {    synchronized (_writeQueue) {      while (_writeQueueMax < _writeQueue.size()) {	try {	  _writeQueue.wait();	} catch (InterruptedException e) {	}      }      _writeQueue.add(block);      _writeQueue.notifyAll();    }  }  //  // management/statistics  //  /**   * The managed name is null   */  public String getName()  {    return null;  }  /**   * The managed type is BlockManager   */  public String getType()  {    return "BlockManager";  }  /**   * Returns the capacity.   */  public long getBlockCapacity()  {    return _blockCache.getCapacity();  }    /**   * Returns the hit count.   */  public long getHitCountTotal()  {    return _blockCache.getHitCount();  }  /**   * Returns the miss count.   */  public long getMissCountTotal()  {    return _blockCache.getMissCount();  }  private static IllegalStateException stateError(String msg)  {    IllegalStateException e = new IllegalStateException(msg);    e.fillInStackTrace();    log.log(Level.WARNING, e.toString(), e);    return e;  }  class BlockManagerWriter implements Runnable {    public void run()    {      while (true) {	try {	  Block block = null;	  	  synchronized (_writeQueue) {	    loop:	    while (true) {	      for (int i = 0; i < _writeQueue.size(); i++) {		block = _writeQueue.get(i);		if (block.isFree())		  break loop;		else		  block = null;	      }	      if (_writeQueue.size() == 0)		_writeQueue.wait();	      else		_writeQueue.wait(10000);	    }	  }	  block.close();	  synchronized (_writeQueue) {	    for (int i = 0; i < _writeQueue.size(); i++) {	      if (block == _writeQueue.get(i)) {		_writeQueue.remove(i);		break;	      }	    }	    _writeQueue.notifyAll();	  }	} catch (InterruptedException e) {	} catch (Throwable e) {	  log.log(Level.WARNING, e.toString(), e);	}      }    }  }}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?