Added vm_delete_aspace() again - unlike before, it now also puts the last
reference to the address space, so that the calling team deletion function doesn't have to do this. delete_address_space() doesn't have to delete any areas anymore, since at the point it's called, there is no area anymore, anyway: each area holds a reference to the address space. git-svn-id: file:///srv/svn/repos/haiku/trunk/current@9276 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
05ab7b5c32
commit
b9d74a6b88
@ -165,10 +165,17 @@ aspace_hash(void *_a, const void *key, uint32 range)
|
||||
}
|
||||
|
||||
|
||||
/** When this function is called, all references to this address space
|
||||
* have been released, so it's safe to remove it.
|
||||
*/
|
||||
|
||||
static void
|
||||
delete_address_space(vm_address_space *aspace)
|
||||
{
|
||||
TRACE(("vm_delete_aspace: called on aspace 0x%lx\n", aspace->id));
|
||||
TRACE(("delete_address_space: called on aspace 0x%lx\n", aspace->id));
|
||||
|
||||
if (aspace == kernel_aspace)
|
||||
panic("tried to delete the kernel aspace!\n");
|
||||
|
||||
// put this aspace in the deletion state
|
||||
// this guarantees that no one else will add regions to the list
|
||||
@ -176,8 +183,6 @@ delete_address_space(vm_address_space *aspace)
|
||||
|
||||
aspace->state = VM_ASPACE_STATE_DELETION;
|
||||
|
||||
vm_delete_areas(aspace);
|
||||
|
||||
(*aspace->translation_map.ops->destroy)(&aspace->translation_map);
|
||||
|
||||
free(aspace->name);
|
||||
@ -244,28 +249,40 @@ vm_get_current_user_aspace_id(void)
|
||||
void
|
||||
vm_put_aspace(vm_address_space *aspace)
|
||||
{
|
||||
// vm_region *region;
|
||||
bool removeit = false;
|
||||
bool remove = false;
|
||||
|
||||
acquire_sem_etc(aspace_hash_sem, WRITE_COUNT, 0, 0);
|
||||
if (atomic_add(&aspace->ref_count, -1) == 1) {
|
||||
hash_remove(aspace_table, aspace);
|
||||
removeit = true;
|
||||
remove = true;
|
||||
}
|
||||
release_sem_etc(aspace_hash_sem, WRITE_COUNT, 0);
|
||||
|
||||
if (!removeit)
|
||||
return;
|
||||
|
||||
TRACE(("vm_put_aspace: reached zero ref, deleting aspace\n"));
|
||||
|
||||
if (aspace == kernel_aspace)
|
||||
panic("vm_put_aspace: tried to delete the kernel aspace!\n");
|
||||
|
||||
if (remove)
|
||||
delete_address_space(aspace);
|
||||
}
|
||||
|
||||
|
||||
/** Deletes all areas in the specified address space, and the address
|
||||
* space by decreasing all reference counters. It also marks the
|
||||
* address space of being in deletion state, so that no more areas
|
||||
* can be created in it.
|
||||
* After this, the address space is not operational anymore, but might
|
||||
* still be in memory until the last reference has been released.
|
||||
*/
|
||||
|
||||
void
|
||||
vm_delete_aspace(vm_address_space *aspace)
|
||||
{
|
||||
acquire_sem_etc(aspace->virtual_map.sem, WRITE_COUNT, 0, 0);
|
||||
aspace->state = VM_ASPACE_STATE_DELETION;
|
||||
release_sem_etc(aspace->virtual_map.sem, WRITE_COUNT, 0);
|
||||
|
||||
vm_delete_areas(aspace);
|
||||
vm_put_aspace(aspace);
|
||||
}
|
||||
|
||||
|
||||
status_t
|
||||
vm_create_aspace(const char *name, addr_t base, addr_t size, bool kernel, vm_address_space **_aspace)
|
||||
{
|
||||
|
Loading…
Reference in New Issue
Block a user