/*
* $Id$
*
* Copyright (C) 2003-2015 JNode.org
*
* This library is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This library 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 Lesser General Public
* License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library; If not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package org.jnode.apps.vmware.disk.handler;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.nio.ByteBuffer;
import java.util.HashMap;
import java.util.Map;
import org.apache.log4j.Logger;
import org.jnode.apps.vmware.disk.descriptor.Descriptor;
import org.jnode.apps.vmware.disk.descriptor.DiskDatabase;
import org.jnode.apps.vmware.disk.extent.Extent;
/**
* Wrote from the 'Virtual Disk Format 1.0' specifications (from VMWare).
*
* @author Fabien DUMINY (fduminy at jnode dot org)
*
*/
public abstract class IOHandler {
private static final Logger LOG = Logger.getLogger(IOHandler.class);
/**
*
*/
public static final int SECTOR_SIZE = 512;
protected static final boolean READ = true;
protected static final boolean WRITE = false;
protected final Descriptor descriptor;
private final long nbSectors;
private final Map<Extent, ExtentIO> extentIOCache = new HashMap<Extent, ExtentIO>();
protected IOHandler(Descriptor descriptor) throws IOException {
this.descriptor = descriptor;
DiskDatabase ddb = descriptor.getDiskDatabase();
nbSectors = ddb.getCylinders() * ddb.getHeads() * ddb.getSectors();
}
/**
*
* @param sector
* @param data
* @throws IOException
*/
public void write(long sector, ByteBuffer data) throws IOException {
int nbSectors = checkBounds(sector, data);
writeImpl(sector, nbSectors, data);
}
/**
*
* @param sector
* @param data
* @throws IOException
*/
public void read(long sector, ByteBuffer data) throws IOException {
int nbSectors = checkBounds(sector, data);
readImpl(sector, nbSectors, data);
}
/**
*
* @throws IOException
*/
public void flush() throws IOException {
for (ExtentIO io : extentIOCache.values()) {
io.flush();
}
extentIOCache.clear();
}
protected int checkBounds(long sector, ByteBuffer buffer) throws IOException {
int nbSectors = buffer.remaining() / SECTOR_SIZE;
if ((buffer.remaining() % SECTOR_SIZE) != 0) {
nbSectors++;
}
checkBounds(sector);
checkBounds(sector + nbSectors - 1);
return nbSectors;
}
protected void checkBounds(long sector) throws IOException {
if (sector < 0) {
throw new IOException("negative sector (actual:" + sector + ")");
}
if (sector >= this.nbSectors) {
throw new IOException("sector above limit(" + this.nbSectors + ") (actual:" + sector +
")");
}
}
protected ExtentIO getExtentIO(long sector, boolean mode) throws IOException {
Extent extent = getExtent(sector, mode);
return getExtentIO(extent);
}
protected final ExtentIO getExtentIO(Extent extent) throws IOException {
ExtentIO io = extentIOCache.get(extent);
if (io == null) {
RandomAccessFile raf = new RandomAccessFile(extent.getFile(), "rw");
LOG.debug("length for file " + extent.getFileName() + " : " + raf.length());
io = createExtentIO(raf, extent);
extentIOCache.put(extent, io);
}
return io;
}
protected ExtentIO createExtentIO(RandomAccessFile raf, Extent extent) {
return new ExtentIO(raf, extent);
}
protected Extent getExtent(long sector, boolean mode) throws IOException {
Extent handler = null;
for (Extent extent : descriptor.getExtents()) {
// LOG.debug(extent.getFileName()+":
// SizeInSectors="+extent.getSizeInSectors());
if (sector < extent.getSizeInSectors()) {
handler = extent;
break;
}
sector -= extent.getSizeInSectors();
}
return handler;
}
/**
*
* @param sector
* @param nbSectors
* @param dst
* @throws IOException
*/
public void readImpl(long sector, int nbSectors, ByteBuffer dst) throws IOException {
LOG.debug("readImpl: sector=" + sector + " nbSectors=" + nbSectors + " buffer.remaining=" +
dst.remaining());
for (int i = 0; i < nbSectors; i++, sector++) {
final ExtentIO io = getExtentIO(sector, READ);
dst.limit(dst.position() + SECTOR_SIZE);
io.read(sector, dst);
}
}
/**
*
* @param sector
* @param nbSectors
* @param src
* @throws IOException
*/
public void writeImpl(long sector, int nbSectors, ByteBuffer src) throws IOException {
LOG.debug("writeImpl: sector=" + sector + " nbSectors=" + nbSectors + " buffer.remaining=" +
src.remaining());
for (int i = 0; i < nbSectors; i++, sector++) {
final ExtentIO io = getExtentIO(sector, WRITE);
src.limit(src.position() + SECTOR_SIZE);
io.write(sector, src);
}
}
/**
*
* @return
*/
public long getNbSectors() {
return nbSectors;
}
}