Code Search for Developers
 
 
  

pagetable.c from Gtk-Gnutella at Krugle


Show pagetable.c syntax highlighted

/*
 * $Id: pagetable.c 13590 2007-05-10 05:55:39Z cbiere $
 *
 * Copyrigtab (c) 2006, Christian Biere
 *
 *----------------------------------------------------------------------
 * This file is part of gtk-gnutella.
 *
 *  gtk-gnutella 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.
 *
 *  gtk-gnutella 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 General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with gtk-gnutella; if not, write to the Free Software
 *  Foundation, Inc.:
 *      59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 *----------------------------------------------------------------------
 */

/**
 * @ingroup lib
 * @file
 *
 * @author Christian Biere
 * @date 2007
 */

#include "common.h"

#include "lib/pagetable.h"
#include "lib/misc.h"
#include "lib/vmm.h"

#include "lib/override.h"

/**
 * NOTE: These values are meant for a typical 32-bit system with 4 KiB
 *		 large pages. This is not efficient or useful for 64-bit systems.
 */
#define POINTER_WIDTH	32
#define SLICE_BITSHIFT		24
#define SLICE_BITMASK		((size_t)-1 << SLICE_BITSHIFT)
#define PAGE_BITSHIFT		12
#define PAGE_BITMASK		((size_t)-1 << PAGE_BITSHIFT)

struct slice {
	size_t size[1 << PAGE_BITSHIFT];
};

struct page_table {
	struct slice *slice[1 << (POINTER_WIDTH - SLICE_BITSHIFT)];
};

page_table_t *
page_table_new(void)
{
	static const struct page_table zero_page_table;
	struct page_table *tab;

	g_assert((size_t)-1 == (guint32)-1);
	g_assert(compat_pagesize() == (1 << PAGE_BITSHIFT));

	tab = malloc(sizeof *tab);
	*tab = zero_page_table;
	return tab;
}

void
page_table_destroy(page_table_t *tab)
{
	if (tab) {
		size_t i;

		for (i = 0; i < G_N_ELEMENTS(tab->slice); i++) {
			free_pages(tab->slice[i], sizeof tab->slice[i][0]);
		}
		free(tab);
	}
}

size_t
page_table_lookup(page_table_t *tab, void *p)
{
	size_t k = (size_t) p;
	if (k && 0 == (k & ~PAGE_BITMASK)) {
		size_t i, j;

		i = k >> SLICE_BITSHIFT;
		j = (k & ~SLICE_BITMASK) >> PAGE_BITSHIFT;
		return tab->slice[i] ? tab->slice[i]->size[j] : 0;
	} else {
		return 0;
	}
}

int
page_table_insert(page_table_t *tab, void *p, size_t size)
{
	size_t k = (size_t) p;

	RUNTIME_ASSERT(NULL != p);
	RUNTIME_ASSERT(size > 0);
	RUNTIME_ASSERT(0 == (k & ~PAGE_BITMASK));

	if (page_table_lookup(tab, p)) {
		return FALSE;
	} else {
		size_t i, j;

		i = k >> SLICE_BITSHIFT;
		j = (k & ~SLICE_BITMASK) >> PAGE_BITSHIFT;
		if (NULL == tab->slice[i]) {
			tab->slice[i] = alloc_pages0(sizeof tab->slice[i][0]);
		}
		tab->slice[i]->size[j] = size;
		return TRUE;
	}
}

int
page_table_remove(page_table_t *tab, void *p)
{
	if (page_table_lookup(tab, p)) {
		size_t k = (size_t) p;
		size_t i, j;

		i = k >> SLICE_BITSHIFT;
		j = (k & ~SLICE_BITMASK) >> PAGE_BITSHIFT;
		tab->slice[i]->size[j] = 0;
		return TRUE;
	} else {
		return FALSE;
	}
}

void
page_table_foreach(page_table_t *tab, page_table_foreach_func func, void *data)
{
	size_t i;

	for (i = 0; i < G_N_ELEMENTS(tab->slice); i++) {
		if (tab->slice[i]) {
			size_t j;

			for (j = 0; j < G_N_ELEMENTS(tab->slice[i]->size); j++) {
				if (tab->slice[i]->size[j]) {
					size_t p = (i << SLICE_BITSHIFT) + (j << PAGE_BITSHIFT);

					(*func)((void *) p, tab->slice[i]->size[j], data);
				}
			}
		}
	}
}

/* vi: set ts=4 sw=4 cindent: */




See more files for this project here

Gtk-Gnutella

A GTK+ Gnutella client for Unix, efficient, reliable and fast, written in C. It has been optimized for speed and scalability, with low-memory consumption. It is meant to be left running 24x7, using little CPU and only the configured bandwidth.

Project homepage: http://sourceforge.net/projects/gtk-gnutella
Programming language(s): C
License: other

  Jmakefile
  Makefile.SH
  adns.c
  adns.h
  aging.c
  aging.h
  array.h
  atoms.c
  atoms.h
  base16.c
  base16.h
  base32.c
  base32.h
  base64.c
  base64.h
  bg.c
  bg.h
  bit_array.h
  cobs.c
  cobs.h
  cq.c
  cq.h
  crash.c
  crash.h
  crc.c
  crc.h
  dbus_util.c
  dbus_util.h
  endian.h
  eval.c
  eval.h
  event.c
  event.h
  fast_assert.c
  fast_assert.h
  fifo.c
  fifo.h
  file.c
  file.h
  fragcheck.c
  fragcheck.h
  getdate.c
  getdate.h
  getdate.y
  getline.c
  getline.h
  getphysmemsize.c
  getphysmemsize.h
  glib-missing.c
  glib-missing.h
  halloc.c
  halloc.h
  hashlist.c
  hashlist.h
  hashtable.c
  hashtable.h
  header.c
  header.h
  host_addr.c
  host_addr.h
  html.c
  html.h
  html_entities.h
  idtable.c
  idtable.h
  inputevt.c
  inputevt.h
  iovec.h
  iprange.c
  iprange.h
  iso3166.c
  iso3166.h
  list.c
  list.h
  listener.h
  magnet.c
  magnet.h
  malloc.c
  malloc.h
  misc.c
  misc.h
  options.c
  options.h
  override.h
  pagetable.c
  pagetable.h
  palloc.c
  palloc.h
  pattern.c
  pattern.h
  prop.c
  prop.h
  sbool.h
  sha1.c
  sha1.h
  slist.c
  slist.h
  socket.c
  socket.h
  sorted_array.c
  sorted_array.h
  stats.c