blob: 3ed800ac2151d5c61a019831b78ff0889ce173e7 [file] [log] [blame]
/*
* 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.ignite.internal.mem.file;
import java.io.Closeable;
import java.io.File;
import java.io.IOException;
import java.io.RandomAccessFile;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import org.apache.ignite.internal.mem.DirectMemoryRegion;
import org.apache.ignite.internal.mem.UnsafeChunk;
import org.apache.ignite.internal.util.typedef.internal.U;
import sun.nio.ch.FileChannelImpl;
/** */
public class MappedFile implements Closeable, DirectMemoryRegion {
/** */
private static final Method map0 = U.findNonPublicMethod(FileChannelImpl.class, "map0", int.class, long.class, long.class);
/** */
private static final Method unmap0 = U.findNonPublicMethod(FileChannelImpl.class, "unmap0", long.class, long.class);
/** */
public static final int MAP_RW = 1;
/** */
private final RandomAccessFile file;
/** */
private final long addr;
/** */
private final long size;
/**
* @param name File path.
* @param size Expected file size. If size is 0, the existing size will be used.
* @throws IOException If failed to open the file or memory-map it.
*/
public MappedFile(File name, long size) throws IOException {
file = new RandomAccessFile(name, "rw");
try {
if (size == 0)
size = file.length();
else
file.setLength(size);
addr = map(file, MAP_RW, 0, size);
this.size = size;
} catch (IOException e) {
file.close();
throw e;
}
}
/**
* Gets underlying random access file.
*
* @return Random access file.
*/
public final RandomAccessFile file() {
return file;
}
/** {@inheritDoc} */
@Override public void close() throws IOException {
try {
unmap(addr, size);
}
finally {
file.close();
}
}
/** {@inheritDoc} */
@Override public final long address() {
return addr;
}
/** {@inheritDoc} */
@Override public final long size() {
return size;
}
/** {@inheritDoc} */
@Override public DirectMemoryRegion slice(long offset) {
if (offset < 0 || offset >= size)
throw new IllegalArgumentException("Failed to create a memory region slice [ptr=" + U.hexLong(addr) +
", len=" + size + ", offset=" + offset + ']');
return new UnsafeChunk(addr + offset, size - offset);
}
/**
* Maps the given region of the file.
*
* @param f File to map.
* @param mode Mode to map.
* @param start Mapping start offset within the file.
* @param size Size of file to map.
* @return Pointer to the mapped memory region.
* @throws IOException
*/
public static long map(RandomAccessFile f, int mode, long start, long size) throws IOException {
try {
return (Long) map0.invoke(f.getChannel(), mode, start, size);
}
catch (IllegalAccessException e) {
throw new IllegalStateException(e);
}
catch (InvocationTargetException e) {
Throwable target = e.getTargetException();
throw (target instanceof IOException) ? (IOException) target : new IOException(target);
}
}
/**
* Un-maps the given region of the file.
*
* @param addr Previously mapped address to un-map.
* @param size Size of the mapped file.
*/
public static void unmap(long addr, long size) {
try {
unmap0.invoke(null, addr, size);
}
catch (IllegalAccessException e) {
throw new IllegalStateException(e);
}
catch (InvocationTargetException e) {
throw new IllegalStateException(e.getTargetException());
}
}
}