servers/hgfs/hgfs_server => servers/hgfs servers/hgfs/libhgfs => lib/libhgfs servers/rs/service => commands/service drivers/memory/memory_driver => drivers/memory drivers/memory/ramdisk => drivers/ramdisk
		
			
				
	
	
		
			195 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			195 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* This file contains directory entry management and the name lookup hashtable.
 | 
						|
 *
 | 
						|
 * The entry points into this file are:
 | 
						|
 *   init_dentry	initialize the directory entry name lookup hashtable
 | 
						|
 *   lookup_dentry	find an inode based on parent directory and name
 | 
						|
 *   add_dentry		add an inode as directory entry to a parent directory
 | 
						|
 *   del_dentry		delete an inode from its parent directory
 | 
						|
 *
 | 
						|
 * Created:
 | 
						|
 *   April 2009 (D.C. van Moolenbroek)
 | 
						|
 */
 | 
						|
 | 
						|
#include "inc.h"
 | 
						|
 | 
						|
PRIVATE LIST_HEAD(hash_head, inode) hash_table[NUM_HASH_SLOTS];
 | 
						|
 | 
						|
FORWARD _PROTOTYPE( void del_one_dentry, (struct inode *ino)		);
 | 
						|
FORWARD _PROTOTYPE( unsigned int hash_dentry, (struct inode *parent,
 | 
						|
							char *name)	);
 | 
						|
 | 
						|
/*===========================================================================*
 | 
						|
 *				init_dentry				     *
 | 
						|
 *===========================================================================*/
 | 
						|
PUBLIC void init_dentry()
 | 
						|
{
 | 
						|
/* Initialize the names hashtable.
 | 
						|
 */
 | 
						|
  int i;
 | 
						|
 | 
						|
  for (i = 0; i < NUM_HASH_SLOTS; i++)
 | 
						|
	LIST_INIT(&hash_table[i]);
 | 
						|
}
 | 
						|
 | 
						|
/*===========================================================================*
 | 
						|
 *				lookup_dentry				     *
 | 
						|
 *===========================================================================*/
 | 
						|
PUBLIC struct inode *lookup_dentry(parent, name)
 | 
						|
struct inode *parent;
 | 
						|
char *name;
 | 
						|
{
 | 
						|
/* Given a directory inode and a component name, look up the inode associated
 | 
						|
 * with that directory entry. Return the inode (with increased reference
 | 
						|
 * count) if found, or NULL otherwise.
 | 
						|
 */
 | 
						|
  struct inode *ino;
 | 
						|
  unsigned int slot;
 | 
						|
 | 
						|
  assert(IS_DIR(parent));
 | 
						|
 | 
						|
  slot = hash_dentry(parent, name);
 | 
						|
 | 
						|
  LIST_FOREACH(ino, &hash_table[slot], i_hash) {
 | 
						|
	if (compare_name(ino->i_name, name) == TRUE)
 | 
						|
		break;
 | 
						|
  }
 | 
						|
 | 
						|
  if (ino == NULL)
 | 
						|
	return NULL;
 | 
						|
 | 
						|
  get_inode(ino);
 | 
						|
 | 
						|
  return ino;
 | 
						|
}
 | 
						|
 | 
						|
/*===========================================================================*
 | 
						|
 *				add_dentry				     *
 | 
						|
 *===========================================================================*/
 | 
						|
PUBLIC void add_dentry(parent, name, ino)
 | 
						|
struct inode *parent;
 | 
						|
char *name;
 | 
						|
struct inode *ino;
 | 
						|
{
 | 
						|
/* Add an entry to a parent inode, in the form of a new inode, with the given
 | 
						|
 * name. An entry with this name must not already exist.
 | 
						|
 */
 | 
						|
  unsigned int slot;
 | 
						|
 | 
						|
  assert(IS_DIR(parent));
 | 
						|
  assert(parent->i_ref > 0);
 | 
						|
  assert(ino->i_ref > 0);
 | 
						|
  assert(name[0]);
 | 
						|
  assert(strlen(name) <= NAME_MAX);
 | 
						|
 | 
						|
  link_inode(parent, ino);
 | 
						|
 | 
						|
  strcpy(ino->i_name, name);
 | 
						|
 | 
						|
  /* hash_add(ino); */
 | 
						|
  slot = hash_dentry(parent, ino->i_name);
 | 
						|
  LIST_INSERT_HEAD(&hash_table[slot], ino, i_hash);
 | 
						|
}
 | 
						|
 | 
						|
/*===========================================================================*
 | 
						|
 *				del_one_dentry				     *
 | 
						|
 *===========================================================================*/
 | 
						|
PRIVATE void del_one_dentry(ino)
 | 
						|
struct inode *ino;
 | 
						|
{
 | 
						|
/* This inode has become inaccessible by name. Disassociate it from its parent
 | 
						|
 * and remove it from the names hash table.
 | 
						|
 */
 | 
						|
 | 
						|
  /* There can and must be exactly one root inode, so don't delete it! */
 | 
						|
  if (IS_ROOT(ino))
 | 
						|
	return;
 | 
						|
 | 
						|
  /* INUSE -> DELETED, CACHED -> FREE */
 | 
						|
 | 
						|
  /* Remove the entry from the hashtable.
 | 
						|
   * Decrease parent's refcount, possibly adding it to the free list.
 | 
						|
   * Do not touch open handles. Do not add to the free list.
 | 
						|
   */
 | 
						|
 | 
						|
  assert(ino->i_parent != NULL);
 | 
						|
 | 
						|
  /* hash_del(ino); */
 | 
						|
  LIST_REMOVE(ino, i_hash);
 | 
						|
 | 
						|
  ino->i_name[0] = 0;
 | 
						|
 | 
						|
  unlink_inode(ino);
 | 
						|
}
 | 
						|
 | 
						|
/*===========================================================================*
 | 
						|
 *				del_dentry				     *
 | 
						|
 *===========================================================================*/
 | 
						|
PUBLIC void del_dentry(ino)
 | 
						|
struct inode *ino;
 | 
						|
{
 | 
						|
/* Disassociate an inode from its parent, effectively deleting it. Recursively
 | 
						|
 * delete all its children as well, fragmenting the deleted branch into single
 | 
						|
 * inodes.
 | 
						|
 */
 | 
						|
  LIST_HEAD(work_list, inode) work_list;
 | 
						|
  struct inode *child;
 | 
						|
 | 
						|
  del_one_dentry(ino);
 | 
						|
 | 
						|
  /* Quick way out: one directory entry that itself has no children. */
 | 
						|
  if (!HAS_CHILDREN(ino))
 | 
						|
	return;
 | 
						|
 | 
						|
  /* Recursively delete all children of the inode as well.
 | 
						|
   * Iterative version: this is potentially 128 levels deep.
 | 
						|
   */
 | 
						|
 | 
						|
  LIST_INIT(&work_list);
 | 
						|
  LIST_INSERT_HEAD(&work_list, ino, i_next);
 | 
						|
 | 
						|
  do {
 | 
						|
	ino = LIST_FIRST(&work_list);
 | 
						|
	LIST_REMOVE(ino, i_next);
 | 
						|
 | 
						|
	assert(IS_DIR(ino));
 | 
						|
 | 
						|
	while (!LIST_EMPTY(&ino->i_child)) {
 | 
						|
		child = LIST_FIRST(&ino->i_child);
 | 
						|
		LIST_REMOVE(child, i_next);
 | 
						|
 | 
						|
		del_one_dentry(child);
 | 
						|
 | 
						|
		if (HAS_CHILDREN(child))
 | 
						|
			LIST_INSERT_HEAD(&work_list, child, i_next);
 | 
						|
	}
 | 
						|
  } while (!LIST_EMPTY(&work_list));
 | 
						|
}
 | 
						|
 | 
						|
/*===========================================================================*
 | 
						|
 *				hash_dentry				     *
 | 
						|
 *===========================================================================*/
 | 
						|
PRIVATE unsigned int hash_dentry(parent, name)
 | 
						|
struct inode *parent;
 | 
						|
char *name;
 | 
						|
{
 | 
						|
/* Generate a hash value for a given name. Normalize the name first, so that
 | 
						|
 * different variations of the name will result in the same hash value.
 | 
						|
 */
 | 
						|
  unsigned int val;
 | 
						|
  char buf[NAME_MAX+1], *p;
 | 
						|
 | 
						|
  dprintf(("HGFS: hash_dentry for '%s'\n", name));
 | 
						|
 | 
						|
  normalize_name(buf, name);
 | 
						|
 | 
						|
  /* djb2 string hash algorithm, XOR variant */
 | 
						|
  val = 5381;
 | 
						|
  for (p = buf; *p; p++)
 | 
						|
	val = ((val << 5) + val) ^ *p;
 | 
						|
 | 
						|
  /* Mix with inode number: typically, many file names occur in several
 | 
						|
   * different directories.
 | 
						|
   */
 | 
						|
  return (val ^ parent->i_num) % NUM_HASH_SLOTS;
 | 
						|
}
 |