a626bdab77
This introduces VMAddressSpace::FindClosestArea() that can be used to find the closest area to a given address in either direction. This is now trivial and efficient since both kernel and user address spaces use a binary search tree. Using FindClosestArea() getting multiple area infos is sped up dramatically as it removes the need for a linear search from the first area to the one given in the cookie on each successive invocation. Change-Id: I227da87d915f6f3d3ef88bfeb6be5d4c97c3baaa Reviewed-on: https://review.haiku-os.org/c/haiku/+/2840 Reviewed-by: waddlesplash <waddlesplash@gmail.com>
225 lines
5.6 KiB
C++
225 lines
5.6 KiB
C++
/*
|
|
* Copyright 2009-2010, Ingo Weinhold, ingo_weinhold@gmx.de.
|
|
* Copyright 2002-2008, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
|
|
* Distributed under the terms of the MIT License.
|
|
*
|
|
* Copyright 2001-2002, Travis Geiselbrecht. All rights reserved.
|
|
* Distributed under the terms of the NewOS License.
|
|
*/
|
|
#ifndef _KERNEL_VM_VM_ADDRESS_SPACE_H
|
|
#define _KERNEL_VM_VM_ADDRESS_SPACE_H
|
|
|
|
|
|
#include <OS.h>
|
|
|
|
#include <vm/vm_priv.h>
|
|
#include <vm/VMArea.h>
|
|
#include <vm/VMTranslationMap.h>
|
|
|
|
|
|
struct virtual_address_restrictions;
|
|
|
|
|
|
struct VMAddressSpace {
|
|
public:
|
|
class AreaIterator;
|
|
|
|
public:
|
|
VMAddressSpace(team_id id, addr_t base,
|
|
size_t size, const char* name);
|
|
virtual ~VMAddressSpace();
|
|
|
|
static status_t Init();
|
|
|
|
team_id ID() const { return fID; }
|
|
addr_t Base() const { return fBase; }
|
|
addr_t EndAddress() const { return fEndAddress; }
|
|
size_t Size() const { return fEndAddress - fBase + 1; }
|
|
size_t FreeSpace() const { return fFreeSpace; }
|
|
bool IsBeingDeleted() const { return fDeleting; }
|
|
|
|
VMTranslationMap* TranslationMap() { return fTranslationMap; }
|
|
|
|
status_t ReadLock()
|
|
{ return rw_lock_read_lock(&fLock); }
|
|
void ReadUnlock()
|
|
{ rw_lock_read_unlock(&fLock); }
|
|
status_t WriteLock()
|
|
{ return rw_lock_write_lock(&fLock); }
|
|
void WriteUnlock()
|
|
{ rw_lock_write_unlock(&fLock); }
|
|
|
|
int32 RefCount() const
|
|
{ return fRefCount; }
|
|
|
|
inline void Get() { atomic_add(&fRefCount, 1); }
|
|
inline void Put();
|
|
void RemoveAndPut();
|
|
|
|
void IncrementFaultCount()
|
|
{ atomic_add(&fFaultCount, 1); }
|
|
void IncrementChangeCount()
|
|
{ fChangeCount++; }
|
|
|
|
inline bool IsRandomizingEnabled() const
|
|
{ return fRandomizingEnabled; }
|
|
inline void SetRandomizingEnabled(bool enabled)
|
|
{ fRandomizingEnabled = enabled; }
|
|
|
|
inline AreaIterator GetAreaIterator();
|
|
|
|
VMAddressSpace*& HashTableLink() { return fHashTableLink; }
|
|
|
|
virtual status_t InitObject();
|
|
|
|
virtual VMArea* FirstArea() const = 0;
|
|
virtual VMArea* NextArea(VMArea* area) const = 0;
|
|
|
|
virtual VMArea* LookupArea(addr_t address) const = 0;
|
|
virtual VMArea* FindClosestArea(addr_t address, bool less) const
|
|
= 0;
|
|
virtual VMArea* CreateArea(const char* name, uint32 wiring,
|
|
uint32 protection,
|
|
uint32 allocationFlags) = 0;
|
|
virtual void DeleteArea(VMArea* area,
|
|
uint32 allocationFlags) = 0;
|
|
virtual status_t InsertArea(VMArea* area, size_t size,
|
|
const virtual_address_restrictions*
|
|
addressRestrictions,
|
|
uint32 allocationFlags, void** _address)
|
|
= 0;
|
|
virtual void RemoveArea(VMArea* area,
|
|
uint32 allocationFlags) = 0;
|
|
|
|
virtual bool CanResizeArea(VMArea* area, size_t newSize) = 0;
|
|
virtual status_t ResizeArea(VMArea* area, size_t newSize,
|
|
uint32 allocationFlags) = 0;
|
|
virtual status_t ShrinkAreaHead(VMArea* area, size_t newSize,
|
|
uint32 allocationFlags) = 0;
|
|
virtual status_t ShrinkAreaTail(VMArea* area, size_t newSize,
|
|
uint32 allocationFlags) = 0;
|
|
|
|
virtual status_t ReserveAddressRange(size_t size,
|
|
const virtual_address_restrictions*
|
|
addressRestrictions,
|
|
uint32 flags, uint32 allocationFlags,
|
|
void** _address) = 0;
|
|
virtual status_t UnreserveAddressRange(addr_t address,
|
|
size_t size, uint32 allocationFlags) = 0;
|
|
virtual void UnreserveAllAddressRanges(
|
|
uint32 allocationFlags) = 0;
|
|
|
|
virtual void Dump() const;
|
|
|
|
static status_t Create(team_id teamID, addr_t base, size_t size,
|
|
bool kernel,
|
|
VMAddressSpace** _addressSpace);
|
|
|
|
static team_id KernelID()
|
|
{ return sKernelAddressSpace->ID(); }
|
|
static VMAddressSpace* Kernel()
|
|
{ return sKernelAddressSpace; }
|
|
static VMAddressSpace* GetKernel();
|
|
|
|
static team_id CurrentID();
|
|
static VMAddressSpace* GetCurrent();
|
|
|
|
static VMAddressSpace* Get(team_id teamID);
|
|
|
|
static VMAddressSpace* DebugFirst();
|
|
static VMAddressSpace* DebugNext(VMAddressSpace* addressSpace);
|
|
static VMAddressSpace* DebugGet(team_id teamID);
|
|
|
|
protected:
|
|
static void _DeleteIfUnreferenced(team_id id);
|
|
|
|
static int _DumpCommand(int argc, char** argv);
|
|
static int _DumpListCommand(int argc, char** argv);
|
|
|
|
protected:
|
|
struct HashDefinition;
|
|
|
|
protected:
|
|
VMAddressSpace* fHashTableLink;
|
|
addr_t fBase;
|
|
addr_t fEndAddress; // base + (size - 1)
|
|
size_t fFreeSpace;
|
|
rw_lock fLock;
|
|
team_id fID;
|
|
int32 fRefCount;
|
|
int32 fFaultCount;
|
|
int32 fChangeCount;
|
|
VMTranslationMap* fTranslationMap;
|
|
bool fRandomizingEnabled;
|
|
bool fDeleting;
|
|
static VMAddressSpace* sKernelAddressSpace;
|
|
};
|
|
|
|
|
|
void
|
|
VMAddressSpace::Put()
|
|
{
|
|
team_id id = fID;
|
|
if (atomic_add(&fRefCount, -1) == 1)
|
|
_DeleteIfUnreferenced(id);
|
|
}
|
|
|
|
|
|
class VMAddressSpace::AreaIterator {
|
|
public:
|
|
AreaIterator()
|
|
{
|
|
}
|
|
|
|
AreaIterator(VMAddressSpace* addressSpace)
|
|
:
|
|
fAddressSpace(addressSpace),
|
|
fNext(addressSpace->FirstArea())
|
|
{
|
|
}
|
|
|
|
bool HasNext() const
|
|
{
|
|
return fNext != NULL;
|
|
}
|
|
|
|
VMArea* Next()
|
|
{
|
|
VMArea* result = fNext;
|
|
if (fNext != NULL)
|
|
fNext = fAddressSpace->NextArea(fNext);
|
|
return result;
|
|
}
|
|
|
|
void Rewind()
|
|
{
|
|
fNext = fAddressSpace->FirstArea();
|
|
}
|
|
|
|
private:
|
|
VMAddressSpace* fAddressSpace;
|
|
VMArea* fNext;
|
|
};
|
|
|
|
|
|
inline VMAddressSpace::AreaIterator
|
|
VMAddressSpace::GetAreaIterator()
|
|
{
|
|
return AreaIterator(this);
|
|
}
|
|
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
void vm_delete_areas(struct VMAddressSpace *aspace, bool deletingAddressSpace);
|
|
#define vm_swap_address_space(from, to) arch_vm_aspace_swap(from, to)
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
|
|
#endif /* _KERNEL_VM_VM_ADDRESS_SPACE_H */
|