HDFS-4124. Refactor INodeDirectory#getExistingPathINodes() to enable returningmore than INode array. Contributed by Jing Zhao.
git-svn-id: https://svn.apache.org/repos/asf/hadoop/common/trunk@1403304 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
parent
4bcf516d0e
commit
1b3b09d947
@ -155,6 +155,9 @@ Trunk (Unreleased)
|
|||||||
HDFS-4122. Cleanup HDFS logs and reduce the size of logged messages.
|
HDFS-4122. Cleanup HDFS logs and reduce the size of logged messages.
|
||||||
(suresh)
|
(suresh)
|
||||||
|
|
||||||
|
HDFS-4124. Refactor INodeDirectory#getExistingPathINodes() to enable
|
||||||
|
returningmore than INode array. (Jing Zhao via suresh)
|
||||||
|
|
||||||
OPTIMIZATIONS
|
OPTIMIZATIONS
|
||||||
|
|
||||||
BUG FIXES
|
BUG FIXES
|
||||||
|
@ -44,10 +44,10 @@
|
|||||||
import org.apache.hadoop.hdfs.protocol.Block;
|
import org.apache.hadoop.hdfs.protocol.Block;
|
||||||
import org.apache.hadoop.hdfs.protocol.ClientProtocol;
|
import org.apache.hadoop.hdfs.protocol.ClientProtocol;
|
||||||
import org.apache.hadoop.hdfs.protocol.DirectoryListing;
|
import org.apache.hadoop.hdfs.protocol.DirectoryListing;
|
||||||
import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
|
||||||
import org.apache.hadoop.hdfs.protocol.FSLimitException;
|
import org.apache.hadoop.hdfs.protocol.FSLimitException;
|
||||||
import org.apache.hadoop.hdfs.protocol.FSLimitException.MaxDirectoryItemsExceededException;
|
import org.apache.hadoop.hdfs.protocol.FSLimitException.MaxDirectoryItemsExceededException;
|
||||||
import org.apache.hadoop.hdfs.protocol.FSLimitException.PathComponentTooLongException;
|
import org.apache.hadoop.hdfs.protocol.FSLimitException.PathComponentTooLongException;
|
||||||
|
import org.apache.hadoop.hdfs.protocol.HdfsConstants;
|
||||||
import org.apache.hadoop.hdfs.protocol.HdfsFileStatus;
|
import org.apache.hadoop.hdfs.protocol.HdfsFileStatus;
|
||||||
import org.apache.hadoop.hdfs.protocol.HdfsLocatedFileStatus;
|
import org.apache.hadoop.hdfs.protocol.HdfsLocatedFileStatus;
|
||||||
import org.apache.hadoop.hdfs.protocol.LocatedBlocks;
|
import org.apache.hadoop.hdfs.protocol.LocatedBlocks;
|
||||||
@ -57,6 +57,7 @@
|
|||||||
import org.apache.hadoop.hdfs.server.blockmanagement.BlockManager;
|
import org.apache.hadoop.hdfs.server.blockmanagement.BlockManager;
|
||||||
import org.apache.hadoop.hdfs.server.blockmanagement.DatanodeDescriptor;
|
import org.apache.hadoop.hdfs.server.blockmanagement.DatanodeDescriptor;
|
||||||
import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.BlockUCState;
|
import org.apache.hadoop.hdfs.server.common.HdfsServerConstants.BlockUCState;
|
||||||
|
import org.apache.hadoop.hdfs.server.namenode.INodeDirectory.INodesInPath;
|
||||||
import org.apache.hadoop.hdfs.util.ByteArray;
|
import org.apache.hadoop.hdfs.util.ByteArray;
|
||||||
|
|
||||||
import com.google.common.base.Preconditions;
|
import com.google.common.base.Preconditions;
|
||||||
@ -548,8 +549,9 @@ boolean unprotectedRenameTo(String src, String dst, long timestamp)
|
|||||||
}
|
}
|
||||||
|
|
||||||
byte[][] dstComponents = INode.getPathComponents(dst);
|
byte[][] dstComponents = INode.getPathComponents(dst);
|
||||||
INode[] dstInodes = new INode[dstComponents.length];
|
INodesInPath dstInodesInPath = rootDir.getExistingPathINodes(dstComponents,
|
||||||
rootDir.getExistingPathINodes(dstComponents, dstInodes, false);
|
dstComponents.length, false);
|
||||||
|
INode[] dstInodes = dstInodesInPath.getINodes();
|
||||||
if (dstInodes[dstInodes.length-1] != null) {
|
if (dstInodes[dstInodes.length-1] != null) {
|
||||||
NameNode.stateChangeLog.warn("DIR* FSDirectory.unprotectedRenameTo: "
|
NameNode.stateChangeLog.warn("DIR* FSDirectory.unprotectedRenameTo: "
|
||||||
+"failed to rename "+src+" to "+dst+
|
+"failed to rename "+src+" to "+dst+
|
||||||
@ -564,7 +566,7 @@ boolean unprotectedRenameTo(String src, String dst, long timestamp)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Ensure dst has quota to accommodate rename
|
// Ensure dst has quota to accommodate rename
|
||||||
verifyQuotaForRename(srcInodes,dstInodes);
|
verifyQuotaForRename(srcInodes, dstInodes);
|
||||||
|
|
||||||
INode dstChild = null;
|
INode dstChild = null;
|
||||||
INode srcChild = null;
|
INode srcChild = null;
|
||||||
@ -668,8 +670,9 @@ boolean unprotectedRenameTo(String src, String dst, long timestamp,
|
|||||||
throw new IOException(error);
|
throw new IOException(error);
|
||||||
}
|
}
|
||||||
final byte[][] dstComponents = INode.getPathComponents(dst);
|
final byte[][] dstComponents = INode.getPathComponents(dst);
|
||||||
final INode[] dstInodes = new INode[dstComponents.length];
|
INodesInPath dstInodesInPath = rootDir.getExistingPathINodes(dstComponents,
|
||||||
rootDir.getExistingPathINodes(dstComponents, dstInodes, false);
|
dstComponents.length, false);
|
||||||
|
final INode[] dstInodes = dstInodesInPath.getINodes();
|
||||||
INode dstInode = dstInodes[dstInodes.length - 1];
|
INode dstInode = dstInodes[dstInodes.length - 1];
|
||||||
if (dstInodes.length == 1) {
|
if (dstInodes.length == 1) {
|
||||||
error = "rename destination cannot be the root";
|
error = "rename destination cannot be the root";
|
||||||
@ -1443,12 +1446,13 @@ boolean mkdirs(String src, PermissionStatus permissions,
|
|||||||
src = normalizePath(src);
|
src = normalizePath(src);
|
||||||
String[] names = INode.getPathNames(src);
|
String[] names = INode.getPathNames(src);
|
||||||
byte[][] components = INode.getPathComponents(names);
|
byte[][] components = INode.getPathComponents(names);
|
||||||
INode[] inodes = new INode[components.length];
|
final int lastInodeIndex = components.length - 1;
|
||||||
final int lastInodeIndex = inodes.length - 1;
|
|
||||||
|
|
||||||
writeLock();
|
writeLock();
|
||||||
try {
|
try {
|
||||||
rootDir.getExistingPathINodes(components, inodes, false);
|
INodesInPath inodesInPath = rootDir.getExistingPathINodes(components,
|
||||||
|
components.length, false);
|
||||||
|
INode[] inodes = inodesInPath.getINodes();
|
||||||
|
|
||||||
// find the index of the first null in inodes[]
|
// find the index of the first null in inodes[]
|
||||||
StringBuilder pathbuilder = new StringBuilder();
|
StringBuilder pathbuilder = new StringBuilder();
|
||||||
@ -1518,16 +1522,14 @@ boolean mkdirs(String src, PermissionStatus permissions,
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
*/
|
|
||||||
INode unprotectedMkdir(String src, PermissionStatus permissions,
|
INode unprotectedMkdir(String src, PermissionStatus permissions,
|
||||||
long timestamp) throws QuotaExceededException,
|
long timestamp) throws QuotaExceededException,
|
||||||
UnresolvedLinkException {
|
UnresolvedLinkException {
|
||||||
assert hasWriteLock();
|
assert hasWriteLock();
|
||||||
byte[][] components = INode.getPathComponents(src);
|
byte[][] components = INode.getPathComponents(src);
|
||||||
INode[] inodes = new INode[components.length];
|
INodesInPath inodesInPath = rootDir.getExistingPathINodes(components,
|
||||||
|
components.length, false);
|
||||||
rootDir.getExistingPathINodes(components, inodes, false);
|
INode[] inodes = inodesInPath.getINodes();
|
||||||
unprotectedMkdir(inodes, inodes.length-1, components[inodes.length-1],
|
unprotectedMkdir(inodes, inodes.length-1, components[inodes.length-1],
|
||||||
permissions, timestamp);
|
permissions, timestamp);
|
||||||
return inodes[inodes.length-1];
|
return inodes[inodes.length-1];
|
||||||
@ -1556,10 +1558,11 @@ private <T extends INode> T addNode(String src, T child,
|
|||||||
byte[] path = components[components.length-1];
|
byte[] path = components[components.length-1];
|
||||||
child.setLocalName(path);
|
child.setLocalName(path);
|
||||||
cacheName(child);
|
cacheName(child);
|
||||||
INode[] inodes = new INode[components.length];
|
|
||||||
writeLock();
|
writeLock();
|
||||||
try {
|
try {
|
||||||
rootDir.getExistingPathINodes(components, inodes, false);
|
INodesInPath inodesInPath = rootDir.getExistingPathINodes(components,
|
||||||
|
components.length, false);
|
||||||
|
INode[] inodes = inodesInPath.getINodes();
|
||||||
return addChild(inodes, inodes.length-1, child, childDiskspace);
|
return addChild(inodes, inodes.length-1, child, childDiskspace);
|
||||||
} finally {
|
} finally {
|
||||||
writeUnlock();
|
writeUnlock();
|
||||||
|
@ -130,9 +130,9 @@ private INode getChildINode(byte[] name) {
|
|||||||
*/
|
*/
|
||||||
private INode getNode(byte[][] components, boolean resolveLink
|
private INode getNode(byte[][] components, boolean resolveLink
|
||||||
) throws UnresolvedLinkException {
|
) throws UnresolvedLinkException {
|
||||||
INode[] inode = new INode[1];
|
INodesInPath inodesInPath = getExistingPathINodes(components, 1,
|
||||||
getExistingPathINodes(components, inode, resolveLink);
|
resolveLink);
|
||||||
return inode[0];
|
return inodesInPath.inodes[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -180,27 +180,29 @@ INode getNode(String path, boolean resolveLink)
|
|||||||
* fill the array with [rootINode,c1,c2,null]
|
* fill the array with [rootINode,c1,c2,null]
|
||||||
*
|
*
|
||||||
* @param components array of path component name
|
* @param components array of path component name
|
||||||
* @param existing array to fill with existing INodes
|
* @param numOfINodes number of INodes to return
|
||||||
* @param resolveLink indicates whether UnresolvedLinkException should
|
* @param resolveLink indicates whether UnresolvedLinkException should
|
||||||
* be thrown when the path refers to a symbolic link.
|
* be thrown when the path refers to a symbolic link.
|
||||||
* @return number of existing INodes in the path
|
* @return the specified number of existing INodes in the path
|
||||||
*/
|
*/
|
||||||
int getExistingPathINodes(byte[][] components, INode[] existing,
|
INodesInPath getExistingPathINodes(byte[][] components, int numOfINodes,
|
||||||
boolean resolveLink) throws UnresolvedLinkException {
|
boolean resolveLink)
|
||||||
|
throws UnresolvedLinkException {
|
||||||
assert this.compareTo(components[0]) == 0 :
|
assert this.compareTo(components[0]) == 0 :
|
||||||
"Incorrect name " + getLocalName() + " expected "
|
"Incorrect name " + getLocalName() + " expected "
|
||||||
+ (components[0] == null? null: DFSUtil.bytes2String(components[0]));
|
+ (components[0] == null? null: DFSUtil.bytes2String(components[0]));
|
||||||
|
|
||||||
|
INodesInPath existing = new INodesInPath(numOfINodes);
|
||||||
INode curNode = this;
|
INode curNode = this;
|
||||||
int count = 0;
|
int count = 0;
|
||||||
int index = existing.length - components.length;
|
int index = numOfINodes - components.length;
|
||||||
if (index > 0) {
|
if (index > 0) {
|
||||||
index = 0;
|
index = 0;
|
||||||
}
|
}
|
||||||
while (count < components.length && curNode != null) {
|
while (count < components.length && curNode != null) {
|
||||||
final boolean lastComp = (count == components.length - 1);
|
final boolean lastComp = (count == components.length - 1);
|
||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
existing[index] = curNode;
|
existing.inodes[index] = curNode;
|
||||||
}
|
}
|
||||||
if (curNode.isLink() && (!lastComp || (lastComp && resolveLink))) {
|
if (curNode.isLink() && (!lastComp || (lastComp && resolveLink))) {
|
||||||
final String path = constructPath(components, 0, components.length);
|
final String path = constructPath(components, 0, components.length);
|
||||||
@ -225,7 +227,7 @@ int getExistingPathINodes(byte[][] components, INode[] existing,
|
|||||||
count++;
|
count++;
|
||||||
index++;
|
index++;
|
||||||
}
|
}
|
||||||
return count;
|
return existing;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -246,11 +248,9 @@ int getExistingPathINodes(byte[][] components, INode[] existing,
|
|||||||
INode[] getExistingPathINodes(String path, boolean resolveLink)
|
INode[] getExistingPathINodes(String path, boolean resolveLink)
|
||||||
throws UnresolvedLinkException {
|
throws UnresolvedLinkException {
|
||||||
byte[][] components = getPathComponents(path);
|
byte[][] components = getPathComponents(path);
|
||||||
INode[] inodes = new INode[components.length];
|
INodesInPath inodes = this.getExistingPathINodes(components,
|
||||||
|
components.length, resolveLink);
|
||||||
this.getExistingPathINodes(components, inodes, resolveLink);
|
return inodes.inodes;
|
||||||
|
|
||||||
return inodes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -341,9 +341,8 @@ INodeDirectory getParent(byte[][] pathComponents
|
|||||||
if (pathComponents.length < 2) // add root
|
if (pathComponents.length < 2) // add root
|
||||||
return null;
|
return null;
|
||||||
// Gets the parent INode
|
// Gets the parent INode
|
||||||
INode[] inodes = new INode[2];
|
INodesInPath inodes = getExistingPathINodes(pathComponents, 2, false);
|
||||||
getExistingPathINodes(pathComponents, inodes, false);
|
INode inode = inodes.inodes[0];
|
||||||
INode inode = inodes[0];
|
|
||||||
if (inode == null) {
|
if (inode == null) {
|
||||||
throw new FileNotFoundException("Parent path does not exist: "+
|
throw new FileNotFoundException("Parent path does not exist: "+
|
||||||
DFSUtil.byteArray2String(pathComponents));
|
DFSUtil.byteArray2String(pathComponents));
|
||||||
@ -443,4 +442,22 @@ int collectSubtreeBlocksAndClear(List<Block> v) {
|
|||||||
children = null;
|
children = null;
|
||||||
return total;
|
return total;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used by
|
||||||
|
* {@link INodeDirectory#getExistingPathINodes(byte[][], int, boolean)}.
|
||||||
|
* Containing INodes information resolved from a given path.
|
||||||
|
*/
|
||||||
|
static class INodesInPath {
|
||||||
|
private INode[] inodes;
|
||||||
|
|
||||||
|
public INodesInPath(int number) {
|
||||||
|
assert (number >= 0);
|
||||||
|
this.inodes = new INode[number];
|
||||||
|
}
|
||||||
|
|
||||||
|
INode[] getINodes() {
|
||||||
|
return inodes;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user