diff options
author | neal <neal> | 2007-11-16 13:35:00 +0000 |
---|---|---|
committer | neal <neal> | 2007-11-16 13:35:00 +0000 |
commit | e46ff816c662bc8b47dfc00bbe5501dbeffd93bb (patch) | |
tree | 16603b813cf7301b85b58f28217aa188825513d8 /viengoos/cap.c | |
parent | 17b21c229fe9756a2e9ec158b6bdf5c2ca9869a5 (diff) |
2007-11-16 Neal H. Walfield <neal@gnu.org>
* viengoos/Makefile.am: New file based on ../wortel/Makefile.am.
* viengoos/headers.m4: New file.
* viengoos/config.m4: New file based on ../wortel/config.m4.
* viengoos/viengoos.h: New file.
* viengoos/viengoos.c: New file.
* viengoos/activity.h: Likewise.
* viengoos/activity.c: Likewise.
* viengoos/as.h: Likewise.
* viengoos/as.c: Likewise.
* viengoos/cap-lookup.c: Likewise.
* viengoos/cap.h: Likewise.
* viengoos/cap.c: Likewise.
* viengoos/thread.h: New file.
* viengoos/thread.c: New file.
* viengoos/object.h: New file.
* viengoos/object.c: New file.
* viengoos/rm.h: New file.
* viengoos/server.c: New file.
* viengoos/server.h: New file.
* viengoos/zalloc.h: Copied from ../physmem.
* viengoos/zalloc.c: Copied from ../physmem.
Don't include "output.h".
Include <hurd/stddef.h>.
Change uses of min_page_size to PAGESIZE.
* viengoos/memory.h: New file.
* viengoos/memory.c: New file.
* viengoos/sigma0.c: Copy from ../wortel.
* viengoos/sigma0.h: Copy from ../wortel.
Don't include "shutdown.h".
Include <hurd/stddef.h>.
* viengoos/bits.h: Likewise.
* viengoos/panic.c: New file.
* viengoos/debug.c: Likewise.
* viengoos/debug.h: Likewise.
* viengoos/boot-modules.h: Likewise.
* viengoos/boot-modules.c: Likewise.
* viengoos/elf.h: Copied from ../wortel.
* viengoos/loader.c: New file based on ../wortel/loader.c.
* viengoos/loader.h: New file.
* viengoos/multiboot.h: Copied from Grub.
* viengoos/mmap.c: New file based on ../physmem/mmap.c.
* viengoos/malloc-wrap.c: New file based on ../physmem/malloc-wrap.c.
* viengoos/malloc.c: Version 2.8.3 of Doug Lea's malloc.c.
* viengoos/malloc.h: Version 2.8.3 of Doug Lea's malloc.h.
* viengoos/ia32-cmain.c: New file based on ../wortel/ia32-cmain.c.
* viengoos/ia32-crt0.S: Copied from ../wortel.
(STACK_SIZE): Use a 16 page stack.
* viengoos/ia32-output.c: Copied from ../wortel.
* viengoos/ia32-shutdown.c: Likewise.
* viengoos/output.h: New file based on ../wortel/output.h.
Include <stdarg.h>.
(cprintf): New definition.
(output_debug): Don't define.
(debug): Don't define.
* viengoos/output.c: New file based on ../wortel/output.c.
Don't include <stdlib.h>.
(vprintf): New function.
(printf): Implement in terms of vprintf.
* viengoos/output-none.c: Copied from ../wortel.
* viengoos/output-serial.c: Likewise.
* viengoos/output-stdio.c: New file.
* viengoos/output-vga.c: Copied from ../wortel.
* viengoos/shutdown.h: New file based on ../wortel/shutdown.h.
Don't include "output.h".
(panic): Don't define.
(shutdown): Rename from this...
(shutdown_machine): ... to this.
* viengoos/shutdown.c: New file based on ../wortel/shutdown.c.
(reset) [_L4_TEST_ENVIRONMENT]: Call abort.
(halt) [_L4_TEST_ENVIRONMENT]: Call abort.
(shutdown): Rename from this...
(shutdown_machine): ... to this.
* viengoos/t-environment.h: New file based on
../libl4/tests/environment.h.
Protect from multiple inclusion.
Include <hurd/stddef.h>.
Include <string.h>.
Include <l4/stubs.h>.
(program_name): New declaration.
(check_nr): Don't assume that val1 and val2 are _L4_word_t, use
typeof instead.
(main): Call output_init.
* viengoos/t-as.c: New file.
Diffstat (limited to 'viengoos/cap.c')
-rw-r--r-- | viengoos/cap.c | 166 |
1 files changed, 166 insertions, 0 deletions
diff --git a/viengoos/cap.c b/viengoos/cap.c new file mode 100644 index 0000000..f1ea7bb --- /dev/null +++ b/viengoos/cap.c @@ -0,0 +1,166 @@ +/* cap.c - Basic capability framework. + Copyright (C) 2007 Free Software Foundation, Inc. + Written by Neal H. Walfield <neal@gnu.org>. + + This file is part of the GNU Hurd. + + The GNU Hurd 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 3 of the + License, or (at your option) any later version. + + The GNU Hurd 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 this program. If not, see + <http://www.gnu.org/licenses/>. */ + +#include <assert.h> +#include <hurd/stddef.h> + +#include "cap.h" +#include "object.h" +#include "activity.h" + +const int cap_type_num_slots[] = { [cap_void] = 0, + [cap_page] = 0, + [cap_rpage] = 0, + [cap_cappage] = CAPPAGE_SLOTS, + [cap_rcappage] = CAPPAGE_SLOTS, + [cap_folio] = 0, + [cap_activity] = 0, + [cap_activity_control] = 0, + [cap_thread] = THREAD_SLOTS }; + +void +cap_set_object (struct cap *cap, + struct object *object, enum cap_type type) +{ + *cap = object_to_cap (object); + assert (cap->type == type); +} + +struct object * +cap_to_object (struct activity *activity, struct cap *cap) +{ + if (cap->type == cap_void) + return NULL; + + struct object *object = object_find (activity, cap->oid); + if (! object) + return NULL; + + struct object_desc *desc = object_to_object_desc (object); + if (desc->version != cap->version) + { + /* Clear the capability to save the effort of looking up the + object in the future. */ + cap->type = cap_void; + return NULL; + } + + /* If the capability is valid, then the types must match. */ + assert (cap->type == desc->type); + + return object; +} + +void +cap_shootdown (struct activity *activity, struct cap *cap) +{ + /* XXX: A recursive function may not be the best idea here. We are + guaranteed, however, at most 63 nested calls. */ + void doit (struct cap *cap, int remaining) + { + int i; + struct object *object; + + remaining -= CAP_GUARD_BITS (cap); + + switch (cap->type) + { + case cap_page: + case cap_rpage: + if (remaining < PAGESIZE_LOG2) + return; + + /* If the object is not in memory, then it can't be + mapped. */ + object = object_find_soft (activity, cap->oid); + if (! object) + return; + + struct object_desc *desc = object_to_object_desc (object); + if (desc->version != cap->version) + { + /* Clear the capability to save the effort of looking up the + object in the future. */ + cap->type = cap_void; + return; + } + +#ifndef _L4_TEST_ENVIRONMENT + l4_fpage_t fpage = l4_fpage ((l4_word_t) object, PAGESIZE); + fpage = l4_fpage_add_rights (fpage, L4_FPAGE_FULLY_ACCESSIBLE); + l4_unmap_fpage (fpage); +#endif + return; + + case cap_cappage: + case cap_rcappage: + if (remaining < CAP_SUBPAGE_SIZE_LOG2 (cap) + PAGESIZE_LOG2) + return; + + object = cap_to_object (activity, cap); + if (! object) + return; + + remaining -= CAP_SUBPAGE_SIZE_LOG2 (cap); + + for (i = 0; i < CAP_SUBPAGE_SIZE_LOG2 (cap); i ++) + doit (&object->caps[i], remaining); + + return; + + case cap_folio: + if (remaining < FOLIO_OBJECTS_LOG2 + PAGESIZE_LOG2) + return; + + object = cap_to_object (activity, cap); + if (! object) + return; + + struct folio *folio = (struct folio *) object; + struct object_desc *fdesc = object_to_object_desc (object); + oid_t foid = fdesc->oid; + + remaining -= FOLIO_OBJECTS_LOG2; + + for (i = 0; i < FOLIO_OBJECTS_LOG2; i ++) + if (folio->objects[i].type == cap_page + || folio->objects[i].type == cap_rpage + || folio->objects[i].type == cap_cappage + || folio->objects[i].type == cap_rcappage) + { + struct cap cap; + + cap.version = folio->objects[i].version; + cap.type = folio->objects[i].type; + cap.addr_trans = CAP_ADDR_TRANS_VOID; + cap.oid = foid + 1 + i; + + doit (&cap, remaining); + } + + return; + + default: + return; + } + } + + doit (cap, ADDR_BITS); +} |