Mega changes. Using hash tables.
Many bug fixes Some formatting changes. Introduction of vnodeManager - to allow cached, opened and mmapped files to work together in love, peace and harmony. git-svn-id: file:///srv/svn/repos/haiku/trunk/current@2012 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
c773cf76b7
commit
bc6600175c
@ -1,6 +1,9 @@
|
|||||||
SubDir OBOS_TOP src kernel vm2 ;
|
SubDir OBOS_TOP src kernel vm2 ;
|
||||||
|
|
||||||
BinCommand vmTest : error.C area.C areaManager.C cacheManager.C page.C pageManager.C swapFileManager.C test.C vmInterface.C vpage.C areaPool.C vnodePool.C vpagePool.C : root be ;
|
BinCommand vmTest : error.C area.C areaManager.C cacheManager.C page.C pageManager.C swapFileManager.C vmInterface.C vpage.C areaPool.C vnodePool.C vpagePool.C test.C : root be ;
|
||||||
|
BinCommand simpleTest : error.C area.C areaManager.C cacheManager.C page.C pageManager.C swapFileManager.C vmInterface.C vpage.C areaPool.C vnodePool.C vpagePool.C simpleTest.C : root be ;
|
||||||
|
BinCommand hashTest : error.C area.C areaManager.C cacheManager.C page.C pageManager.C swapFileManager.C vmInterface.C vpage.C areaPool.C vnodePool.C vpagePool.C hashTest.C : root be ;
|
||||||
BinCommand olTest : error.C olTest.C : root be ;
|
BinCommand olTest : error.C olTest.C : root be ;
|
||||||
|
BinCommand pmTest : error.C pageManager.C pageManTest.C page.C : root be ;
|
||||||
|
|
||||||
|
|
||||||
|
@ -9,29 +9,24 @@
|
|||||||
|
|
||||||
extern vmHeaderBlock *vmBlock;
|
extern vmHeaderBlock *vmBlock;
|
||||||
|
|
||||||
bool vpageIsLessThan(void *a,void *b)
|
ulong vpageHash (node &vp) {return reinterpret_cast <vpage &>(vp).hash();}
|
||||||
{
|
bool vpageisEqual (node &vp,node &vp2) {return reinterpret_cast <vpage &>(vp)==reinterpret_cast <vpage &>(vp2);}
|
||||||
return (((reinterpret_cast<vpage *>(a))->getStartAddress()) < (reinterpret_cast<vpage *>(b))->getStartAddress());
|
|
||||||
|
area::area(void) : vpages(AREA_HASH_TABLE_SIZE) {
|
||||||
|
vpages.setHash(vpageHash);
|
||||||
|
vpages.setIsEqual(vpageisEqual);
|
||||||
}
|
}
|
||||||
|
|
||||||
area::area(void)
|
void area::setup (areaManager *myManager) {
|
||||||
{
|
|
||||||
vpages.setIsLessThan(vpageIsLessThan);
|
|
||||||
}
|
|
||||||
|
|
||||||
void area::setup (areaManager *myManager)
|
|
||||||
{
|
|
||||||
//error ("area::setup setting up new area\n");
|
//error ("area::setup setting up new area\n");
|
||||||
manager=myManager;
|
manager=myManager;
|
||||||
//error ("area::setup done setting up new area\n");
|
//error ("area::setup done setting up new area\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
unsigned long area::mapAddressSpecToAddress(addressSpec type,void * req,int pageCount)
|
unsigned long area::mapAddressSpecToAddress(addressSpec type,void * req,int pageCount) {
|
||||||
{
|
|
||||||
// We will lock in the callers
|
// We will lock in the callers
|
||||||
unsigned long base,requested=(unsigned long)req;
|
unsigned long base,requested=(unsigned long)req;
|
||||||
switch (type)
|
switch (type) {
|
||||||
{
|
|
||||||
case EXACT:
|
case EXACT:
|
||||||
base=manager->getNextAddress(pageCount,requested);
|
base=manager->getNextAddress(pageCount,requested);
|
||||||
if (base!=requested)
|
if (base!=requested)
|
||||||
@ -54,8 +49,9 @@ unsigned long area::mapAddressSpecToAddress(addressSpec type,void * req,int page
|
|||||||
return base;
|
return base;
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::createAreaGuts( char *inName, int pageCount, void **address, addressSpec type, pageState inState, protectType protect, bool inFinalWrite, int fd, size_t offset, area *originalArea=NULL /* For clone only*/)
|
status_t area::createAreaGuts( char *inName, int pageCount, void **address, addressSpec type, pageState inState, protectType protect, bool inFinalWrite, int fd, size_t offset, area *originalArea=NULL /* For clone only*/) {
|
||||||
{
|
error ("area::createAreaGuts : name = %s, pageCount = %d, address = %ld, addressSpec = %d, pageState = %d, protection = %d, inFinalWrite = %d, fd = %d, offset = %d,originalArea=%ld\n",
|
||||||
|
inName,pageCount,address,type,inState,protect,inFinalWrite,fd,offset,originalArea);
|
||||||
strcpy(name,inName);
|
strcpy(name,inName);
|
||||||
vpage *newPage;
|
vpage *newPage;
|
||||||
|
|
||||||
@ -67,53 +63,50 @@ status_t area::createAreaGuts( char *inName, int pageCount, void **address, addr
|
|||||||
end_address=base+(pageCount*PAGE_SIZE)-1;
|
end_address=base+(pageCount*PAGE_SIZE)-1;
|
||||||
*address=(void *)base;
|
*address=(void *)base;
|
||||||
finalWrite=inFinalWrite;
|
finalWrite=inFinalWrite;
|
||||||
if (!originalArea)
|
if (originalArea==NULL) // Not for cloning
|
||||||
for (int i=0;i<pageCount;i++)
|
for (int i=0;i<pageCount;i++) {
|
||||||
{
|
|
||||||
newPage=new (vmBlock->vpagePool->get()) vpage;
|
newPage=new (vmBlock->vpagePool->get()) vpage;
|
||||||
if (fd)
|
if (fd) {
|
||||||
{
|
// vnode *newVnode=new (vmBlock->vnodePool->get()) vnode;
|
||||||
vnode *newVnode=new (vmBlock->vnodePool->get()) vnode;
|
void *currentMemoryLocation;
|
||||||
|
// A future implementation vnode *newVnode=vmBlock->vnodePool->getNode(fd,offset,PAGE_SIZE*i,true,currentMemoryLocation);
|
||||||
|
vnode *newVnode=vmBlock->vnodePool->get();
|
||||||
newVnode->fd=fd;
|
newVnode->fd=fd;
|
||||||
newVnode->offset=offset+PAGE_SIZE*i;
|
newVnode->offset=offset;
|
||||||
newVnode->valid=true;
|
|
||||||
newPage->setup(base+PAGE_SIZE*i,newVnode,NULL,protect,inState);
|
newPage->setup(base+PAGE_SIZE*i,newVnode,NULL,protect,inState);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
newPage->setup(base+PAGE_SIZE*i,NULL,NULL,protect,inState);
|
newPage->setup(base+PAGE_SIZE*i,NULL,NULL,protect,inState);
|
||||||
vpages.add(newPage);
|
vpages.add(newPage);
|
||||||
}
|
}
|
||||||
else
|
else // cloned
|
||||||
for (struct node *cur=originalArea->vpages.rock;cur;cur=cur->next)
|
// Need to lock other area, here, just in case...
|
||||||
{
|
|
||||||
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
vpage *page=(vpage *)cur;
|
vpage *page=(vpage *)cur;
|
||||||
newPage=new (vmBlock->vpagePool->get()) vpage;
|
newPage=new (vmBlock->vpagePool->get()) vpage;
|
||||||
newPage->setup(base,page->getBacking(),page->getPhysPage(),protect,inState);// Cloned area has the same physical page and backing store...
|
newPage->setup(base,page->getBacking(),page->getPhysPage(),protect,inState);// Cloned area has the same physical page and backing store...
|
||||||
vpages.add(newPage);
|
vpages.add(newPage);
|
||||||
base+=PAGE_SIZE;
|
base+=PAGE_SIZE;
|
||||||
}
|
}
|
||||||
|
dump();
|
||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::createAreaMappingFile(char *inName, int pageCount,void **address, addressSpec type,pageState inState,protectType protect,int fd,size_t offset)
|
status_t area::createAreaMappingFile(char *inName, int pageCount,void **address, addressSpec type,pageState inState,protectType protect,int fd,size_t offset) {
|
||||||
{
|
|
||||||
return createAreaGuts(inName,pageCount,address,type,inState,protect,true,fd,offset);
|
return createAreaGuts(inName,pageCount,address,type,inState,protect,true,fd,offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::createArea(char *inName, int pageCount,void **address, addressSpec type,pageState inState,protectType protect)
|
status_t area::createArea(char *inName, int pageCount,void **address, addressSpec type,pageState inState,protectType protect) {
|
||||||
{
|
|
||||||
return createAreaGuts(inName,pageCount,address,type,inState,protect,false,0,0);
|
return createAreaGuts(inName,pageCount,address,type,inState,protect,false,0,0);
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::cloneArea(area *origArea, char *inName, void **address, addressSpec type,pageState inState,protectType protect)
|
status_t area::cloneArea(area *origArea, char *inName, void **address, addressSpec type,pageState inState,protectType protect) {
|
||||||
{
|
if (type==CLONE) {
|
||||||
if (type==CLONE)
|
|
||||||
{
|
|
||||||
*address=(void *)(origArea->getStartAddress());
|
*address=(void *)(origArea->getStartAddress());
|
||||||
type=EXACT;
|
type=EXACT;
|
||||||
}
|
}
|
||||||
if (origArea->getAreaManager()!=manager)
|
if (origArea->getAreaManager()!=manager) {
|
||||||
{
|
|
||||||
origArea->getAreaManager()->lock(); // This is just begging for a deadlock...
|
origArea->getAreaManager()->lock(); // This is just begging for a deadlock...
|
||||||
status_t retVal = createAreaGuts(inName,origArea->getPageCount(),address,type,inState,protect,false,0,0,origArea);
|
status_t retVal = createAreaGuts(inName,origArea->getPageCount(),address,type,inState,protect,false,0,0,origArea);
|
||||||
origArea->getAreaManager()->unlock();
|
origArea->getAreaManager()->unlock();
|
||||||
@ -123,14 +116,12 @@ status_t area::cloneArea(area *origArea, char *inName, void **address, addressSp
|
|||||||
return createAreaGuts(inName,origArea->getPageCount(),address,type,inState,protect,false,0,0,origArea);
|
return createAreaGuts(inName,origArea->getPageCount(),address,type,inState,protect,false,0,0,origArea);
|
||||||
}
|
}
|
||||||
|
|
||||||
void area::freeArea(void)
|
void area::freeArea(void) {
|
||||||
{
|
|
||||||
//error ("area::freeArea: starting \n");
|
//error ("area::freeArea: starting \n");
|
||||||
|
|
||||||
// vpages.dump();
|
// vpages.dump();
|
||||||
node *cur;
|
node *cur;
|
||||||
while ((cur=vpages.next())!=NULL)
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
{
|
|
||||||
//error ("area::freeArea: wasting a page: %x\n",cur);
|
//error ("area::freeArea: wasting a page: %x\n",cur);
|
||||||
vpage *page=reinterpret_cast<vpage *>(cur);
|
vpage *page=reinterpret_cast<vpage *>(cur);
|
||||||
if (finalWrite)
|
if (finalWrite)
|
||||||
@ -144,8 +135,7 @@ void area::freeArea(void)
|
|||||||
//error ("area::freeArea: ending \n");
|
//error ("area::freeArea: ending \n");
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::getInfo(area_info *dest)
|
status_t area::getInfo(area_info *dest) {
|
||||||
{
|
|
||||||
dest->area=areaID;
|
dest->area=areaID;
|
||||||
strcpy(dest->name,name);
|
strcpy(dest->name,name);
|
||||||
dest->size=end_address-start_address;
|
dest->size=end_address-start_address;
|
||||||
@ -156,8 +146,7 @@ status_t area::getInfo(area_info *dest)
|
|||||||
dest->in_count=0;
|
dest->in_count=0;
|
||||||
dest->out_count=0;
|
dest->out_count=0;
|
||||||
dest->copy_count=0;
|
dest->copy_count=0;
|
||||||
for (struct node *cur=vpages.rock;cur;cur=cur->next)
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
{
|
|
||||||
vpage *page=(vpage *)cur;
|
vpage *page=(vpage *)cur;
|
||||||
if (page->isMapped())
|
if (page->isMapped())
|
||||||
dest->ram_size+=PAGE_SIZE;
|
dest->ram_size+=PAGE_SIZE;
|
||||||
@ -169,40 +158,40 @@ status_t area::getInfo(area_info *dest)
|
|||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool area::contains(void *address)
|
bool area::contains(void *address) {
|
||||||
{
|
|
||||||
unsigned long base=(unsigned long)(address);
|
unsigned long base=(unsigned long)(address);
|
||||||
// error ("area::contains: looking for %d in %d -- %d, value = %d\n",base,start_address,end_address, ((start_address<=base) && (end_address>=base)));
|
// error ("area::contains: looking for %d in %d -- %d, value = %d\n",base,start_address,end_address, ((start_address<=base) && (end_address>=base)));
|
||||||
|
|
||||||
return ((start_address<=base) && (base<=end_address));
|
return ((start_address<=base) && (base<=end_address));
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::resize(size_t newSize)
|
status_t area::resize(size_t newSize) {
|
||||||
{
|
|
||||||
size_t oldSize =end_address-start_address;
|
size_t oldSize =end_address-start_address;
|
||||||
if (newSize==oldSize)
|
if (newSize==oldSize)
|
||||||
return B_OK;
|
return B_OK;
|
||||||
if (newSize>oldSize)
|
if (newSize>oldSize) {
|
||||||
{
|
|
||||||
int pageCount = (newSize - oldSize + PAGE_SIZE - 1) / PAGE_SIZE;
|
int pageCount = (newSize - oldSize + PAGE_SIZE - 1) / PAGE_SIZE;
|
||||||
vpage *newPage;
|
vpage *newPage;
|
||||||
for (int i=0;i<pageCount;i++)
|
for (int i=0;i<pageCount;i++) {
|
||||||
{
|
|
||||||
newPage=new (vmBlock->vpagePool->get()) vpage;
|
newPage=new (vmBlock->vpagePool->get()) vpage;
|
||||||
newPage->setup(end_address+PAGE_SIZE*i-1,NULL,NULL,protection,state);
|
newPage->setup(end_address+PAGE_SIZE*i-1,NULL,NULL,protection,state);
|
||||||
vpages.add(newPage);
|
vpages.add(newPage);
|
||||||
}
|
}
|
||||||
end_address+=start_address+newSize;
|
end_address+=start_address+newSize;
|
||||||
}
|
}
|
||||||
else
|
else {
|
||||||
{
|
|
||||||
int pageCount = (oldSize - newSize + PAGE_SIZE - 1) / PAGE_SIZE;
|
int pageCount = (oldSize - newSize + PAGE_SIZE - 1) / PAGE_SIZE;
|
||||||
vpage *oldPage;
|
vpage *oldPage;
|
||||||
struct node *cur;
|
struct node *cur;
|
||||||
for (int i=0;i<pageCount;i++) // This is probably really slow. Adding an "end" to list would be faster.
|
for (int i=0;i<pageCount;i++) { // This is probably really slow. OTOH, how often do people shrink their allocations?
|
||||||
{
|
node *max;
|
||||||
for (cur=vpages.rock;cur->next;cur=cur->next); // INTENTIONAL - find the last one;
|
void *maxAddress=NULL,*curAddress;
|
||||||
oldPage=(vpage *)cur;
|
for (hashIterate hi(vpages);node *cur=hi.get();)
|
||||||
|
if ((curAddress=(reinterpret_cast<vpage *>(cur))->getStartAddress()) > maxAddress) {
|
||||||
|
maxAddress=curAddress;
|
||||||
|
max=cur;
|
||||||
|
}
|
||||||
|
oldPage=reinterpret_cast<vpage *>(max);
|
||||||
vpages.remove(cur);
|
vpages.remove(cur);
|
||||||
if (finalWrite)
|
if (finalWrite)
|
||||||
oldPage->flush();
|
oldPage->flush();
|
||||||
@ -213,10 +202,8 @@ status_t area::resize(size_t newSize)
|
|||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
status_t area::setProtection(protectType prot)
|
status_t area::setProtection(protectType prot) {
|
||||||
{
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
for (struct node *cur=vpages.rock;cur;cur=cur->next)
|
|
||||||
{
|
|
||||||
vpage *page=(vpage *)cur;
|
vpage *page=(vpage *)cur;
|
||||||
page->setProtection(prot);
|
page->setProtection(prot);
|
||||||
}
|
}
|
||||||
@ -224,21 +211,12 @@ status_t area::setProtection(protectType prot)
|
|||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
vpage *area::findVPage(unsigned long address)
|
vpage *area::findVPage(unsigned long address) {
|
||||||
{
|
vpage findMe(address);
|
||||||
for (struct node *cur=vpages.rock;cur;cur=cur->next)
|
return reinterpret_cast <vpage *>(vpages.find(&findMe));
|
||||||
{
|
|
||||||
vpage *page=(vpage *)cur;
|
|
||||||
//error ("Examining following vpage, looking for address %lx\n",address);
|
|
||||||
//page->dump();
|
|
||||||
if (page->contains(address))
|
|
||||||
return page;
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool area::fault(void *fault_address, bool writeError) // true = OK, false = panic.
|
bool area::fault(void *fault_address, bool writeError) { // true = OK, false = panic.
|
||||||
{
|
|
||||||
vpage *page=findVPage((unsigned long)fault_address);
|
vpage *page=findVPage((unsigned long)fault_address);
|
||||||
if (page)
|
if (page)
|
||||||
return page->fault(fault_address,writeError);
|
return page->fault(fault_address,writeError);
|
||||||
@ -246,8 +224,7 @@ bool area::fault(void *fault_address, bool writeError) // true = OK, false = pan
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
char area::getByte(unsigned long address) // This is for testing only
|
char area::getByte(unsigned long address) { // This is for testing only
|
||||||
{
|
|
||||||
vpage *page=findVPage(address);
|
vpage *page=findVPage(address);
|
||||||
if (page)
|
if (page)
|
||||||
return page->getByte(address,manager);
|
return page->getByte(address,manager);
|
||||||
@ -255,15 +232,13 @@ char area::getByte(unsigned long address) // This is for testing only
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void area::setByte(unsigned long address,char value) // This is for testing only
|
void area::setByte(unsigned long address,char value) { // This is for testing only
|
||||||
{
|
|
||||||
vpage *page=findVPage(address);
|
vpage *page=findVPage(address);
|
||||||
if (page)
|
if (page)
|
||||||
page->setByte(address,value,manager);
|
page->setByte(address,value,manager);
|
||||||
}
|
}
|
||||||
|
|
||||||
int area::getInt(unsigned long address) // This is for testing only
|
int area::getInt(unsigned long address) { // This is for testing only
|
||||||
{
|
|
||||||
vpage *page=findVPage(address);
|
vpage *page=findVPage(address);
|
||||||
if (page)
|
if (page)
|
||||||
return page->getInt(address,manager);
|
return page->getInt(address,manager);
|
||||||
@ -271,36 +246,30 @@ int area::getInt(unsigned long address) // This is for testing only
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void area::setInt(unsigned long address,int value) // This is for testing only
|
void area::setInt(unsigned long address,int value) { // This is for testing only
|
||||||
{
|
|
||||||
vpage *page=findVPage(address);
|
vpage *page=findVPage(address);
|
||||||
if (page)
|
if (page)
|
||||||
page->setInt(address,value,manager);
|
page->setInt(address,value,manager);
|
||||||
}
|
}
|
||||||
|
|
||||||
void area::pager(int desperation)
|
void area::pager(int desperation) {
|
||||||
{
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
for (struct node *cur=vpages.rock;cur;cur=cur->next)
|
|
||||||
{
|
|
||||||
vpage *page=(vpage *)cur;
|
vpage *page=(vpage *)cur;
|
||||||
page->pager(desperation);
|
page->pager(desperation);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void area::saver(void)
|
void area::saver(void) {
|
||||||
{
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
for (struct node *cur=vpages.rock;cur;cur=cur->next)
|
|
||||||
{
|
|
||||||
vpage *page=(vpage *)cur;
|
vpage *page=(vpage *)cur;
|
||||||
page->saver();
|
page->saver();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void area::dump(void)
|
void area::dump(void) {
|
||||||
{
|
|
||||||
error ("area::dump: size = %ld, lock = %d, address = %lx\n",end_address-start_address,state,start_address);
|
error ("area::dump: size = %ld, lock = %d, address = %lx\n",end_address-start_address,state,start_address);
|
||||||
for (struct node *cur=vpages.rock;cur;)
|
vpages.dump();
|
||||||
{
|
for (hashIterate hi(vpages);node *cur=hi.get();) {
|
||||||
vpage *page=(vpage *)cur;
|
vpage *page=(vpage *)cur;
|
||||||
page->dump();
|
page->dump();
|
||||||
cur=cur->next;
|
cur=cur->next;
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
#include "OS.h"
|
#include "OS.h"
|
||||||
#include "vm.h"
|
#include "vm.h"
|
||||||
#include "list.h"
|
#include "list.h"
|
||||||
#include "olist.h"
|
#include "hashTable.h"
|
||||||
|
|
||||||
class areaManager;
|
class areaManager;
|
||||||
class vpage;
|
class vpage;
|
||||||
@ -11,7 +11,7 @@ class vpage;
|
|||||||
class area : public node
|
class area : public node
|
||||||
{
|
{
|
||||||
protected:
|
protected:
|
||||||
orderedList vpages;
|
hashTable vpages;
|
||||||
char name[B_OS_NAME_LENGTH];
|
char name[B_OS_NAME_LENGTH];
|
||||||
pageState state;
|
pageState state;
|
||||||
protectType protection;
|
protectType protection;
|
||||||
@ -25,36 +25,47 @@ class area : public node
|
|||||||
unsigned long end_address;
|
unsigned long end_address;
|
||||||
vpage *findVPage(unsigned long);
|
vpage *findVPage(unsigned long);
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
area(void);
|
area(void);
|
||||||
void setup(areaManager *myManager);
|
void setup(areaManager *myManager);
|
||||||
bool nameMatch(char *matchName) {return (strcmp(matchName,name)==0);}
|
void freeArea(void);
|
||||||
unsigned long mapAddressSpecToAddress(addressSpec type,void *requested,int pageCount);
|
|
||||||
status_t createAreaGuts( char *inName, int pageCount, void **address, addressSpec type, pageState inState, protectType protect, bool inFinalWrite, int fd, size_t offset, area *originalArea=NULL /* For clone only*/);
|
status_t createAreaGuts( char *inName, int pageCount, void **address, addressSpec type, pageState inState, protectType protect, bool inFinalWrite, int fd, size_t offset, area *originalArea=NULL /* For clone only*/);
|
||||||
status_t createAreaMappingFile(char *name, int pageCount,void **address, addressSpec type,pageState state,protectType protect,int fd,size_t offset);
|
status_t createAreaMappingFile(char *name, int pageCount,void **address, addressSpec type,pageState state,protectType protect,int fd,size_t offset);
|
||||||
status_t createArea (char *name, int pageCount,void **address, addressSpec type,pageState state,protectType protect);
|
status_t createArea (char *name, int pageCount,void **address, addressSpec type,pageState state,protectType protect);
|
||||||
status_t cloneArea(area *area, char *inName, void **address, addressSpec type,pageState inState,protectType protect);
|
status_t cloneArea(area *area, char *inName, void **address, addressSpec type,pageState inState,protectType protect);
|
||||||
int getAreaID(void) {return areaID;}
|
unsigned long mapAddressSpecToAddress(addressSpec type,void *requested,int pageCount);
|
||||||
|
|
||||||
|
// Mutators
|
||||||
void setAreaID(int id) {areaID=id;}
|
void setAreaID(int id) {areaID=id;}
|
||||||
void freeArea(void);
|
|
||||||
status_t getInfo(area_info *dest);
|
|
||||||
bool contains(void *address);
|
|
||||||
status_t resize(size_t newSize);
|
|
||||||
status_t setProtection(protectType prot);
|
status_t setProtection(protectType prot);
|
||||||
bool couldAdd(unsigned long start,unsigned long end) { return ((end<start_address) || (start>end_address));}
|
status_t resize(size_t newSize);
|
||||||
unsigned long getEndAddress(void) {return end_address;}
|
|
||||||
unsigned long getStartAddress(void) {return start_address;}
|
// Accessors
|
||||||
void pager(int desperation);
|
status_t getInfo(area_info *dest);
|
||||||
void saver(void);
|
int getAreaID(void) {return areaID;}
|
||||||
unsigned long getSize(void) {return getEndAddress()-getStartAddress();}
|
unsigned long getSize(void) {return getEndAddress()-getStartAddress();}
|
||||||
unsigned long getPageCount(void) {return (getEndAddress()-getStartAddress())/PAGE_SIZE;}
|
unsigned long getPageCount(void) {return (getEndAddress()-getStartAddress())/PAGE_SIZE;}
|
||||||
areaManager *getAreaManager(void) {return manager;}
|
areaManager *getAreaManager(void) {return manager;}
|
||||||
|
unsigned long getEndAddress(void) {return end_address;}
|
||||||
|
unsigned long getStartAddress(void) {return start_address;}
|
||||||
|
|
||||||
|
// Debugging
|
||||||
void dump(void);
|
void dump(void);
|
||||||
bool fault(void *fault_address, bool writeError); // true = OK, false = panic.
|
|
||||||
|
|
||||||
char getByte(unsigned long ); // This is for testing only
|
char getByte(unsigned long ); // This is for testing only
|
||||||
void setByte(unsigned long ,char value); // This is for testing only
|
void setByte(unsigned long ,char value); // This is for testing only
|
||||||
int getInt(unsigned long ); // This is for testing only
|
int getInt(unsigned long ); // This is for testing only
|
||||||
void setInt(unsigned long ,int value); // This is for testing only
|
void setInt(unsigned long ,int value); // This is for testing only
|
||||||
|
|
||||||
|
// Comparisson with others
|
||||||
|
bool nameMatch(char *matchName) {return (strcmp(matchName,name)==0);}
|
||||||
|
bool couldAdd(unsigned long start,unsigned long end) { return ((end<start_address) || (start>end_address));}
|
||||||
|
bool contains(void *address);
|
||||||
|
|
||||||
|
// External methods for "server" type calls
|
||||||
|
void pager(int desperation);
|
||||||
|
void saver(void);
|
||||||
|
bool fault(void *fault_address, bool writeError); // true = OK, false = panic.
|
||||||
|
|
||||||
|
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
@ -142,20 +142,20 @@ int areaManager::createArea(char *AreaName,int pageCount,void **address, address
|
|||||||
error ("Creating an area\n");
|
error ("Creating an area\n");
|
||||||
lock();
|
lock();
|
||||||
area *newArea = new (vmBlock->areaPool->get()) area;
|
area *newArea = new (vmBlock->areaPool->get()) area;
|
||||||
// error ("areaManager::createArea - got a new area (%p) from the areaPool\n",newArea);
|
error ("areaManager::createArea - got a new area (%p) from the areaPool\n",newArea);
|
||||||
newArea->setup(this);
|
newArea->setup(this);
|
||||||
// error ("areaManager::createArea - setup complete\n");
|
error ("areaManager::createArea - setup complete\n");
|
||||||
newArea->createArea(AreaName,pageCount,address,addType,state,protect);
|
newArea->createArea(AreaName,pageCount,address,addType,state,protect);
|
||||||
// error ("areaManager::createArea - new area's createArea called\n");
|
error ("areaManager::createArea - new area's createArea called\n");
|
||||||
atomic_add(&nextAreaID,1);
|
atomic_add(&nextAreaID,1);
|
||||||
newArea->setAreaID(nextAreaID);
|
newArea->setAreaID(nextAreaID);
|
||||||
// error ("areaManager::createArea - new area's setAreaID called\n");
|
error ("areaManager::createArea - new area's setAreaID called\n");
|
||||||
addArea(newArea);
|
addArea(newArea);
|
||||||
// error ("areaManager::createArea - new area added to list\n");
|
error ("areaManager::createArea - new area added to list\n");
|
||||||
int retVal=newArea->getAreaID();
|
int retVal=newArea->getAreaID();
|
||||||
// error ("areaManager::createArea - new area id found\n");
|
error ("areaManager::createArea - new area id found\n");
|
||||||
unlock();
|
unlock();
|
||||||
//error ("Done Creating an area\n");
|
error ("Done Creating an area\n");
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -184,13 +184,18 @@ int areaManager::cloneArea(int newAreaID,char *AreaName,void **address, addressS
|
|||||||
char areaManager::getByte(unsigned long address)
|
char areaManager::getByte(unsigned long address)
|
||||||
{
|
{
|
||||||
area *myArea;
|
area *myArea;
|
||||||
|
// error ("areaManager::getByte : starting\n");
|
||||||
int retVal;
|
int retVal;
|
||||||
lock();
|
lock();
|
||||||
myArea=findArea((void *)address);
|
myArea=findArea((void *)address);
|
||||||
if (myArea)
|
if (myArea)
|
||||||
retVal=myArea->getByte(address);
|
retVal=myArea->getByte(address);
|
||||||
else
|
else
|
||||||
retVal= 0;
|
{
|
||||||
|
char temp[1000];
|
||||||
|
sprintf (temp,"Unable to find an area for address %d\n",address);
|
||||||
|
throw (temp);
|
||||||
|
}
|
||||||
unlock();
|
unlock();
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
@ -204,18 +209,29 @@ int areaManager::getInt(unsigned long address)
|
|||||||
if (myArea)
|
if (myArea)
|
||||||
retVal=myArea->getInt(address);
|
retVal=myArea->getInt(address);
|
||||||
else
|
else
|
||||||
retVal= 0;
|
{
|
||||||
|
char temp[1000];
|
||||||
|
sprintf (temp,"Unable to find an area for address %d\n",address);
|
||||||
|
throw (temp);
|
||||||
|
}
|
||||||
unlock();
|
unlock();
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
void areaManager::setByte(unsigned long address,char value)
|
void areaManager::setByte(unsigned long address,char value)
|
||||||
{
|
{
|
||||||
|
// error ("areaManager::setByte : starting\n");
|
||||||
area *myArea;
|
area *myArea;
|
||||||
lock();
|
lock();
|
||||||
myArea=findArea((void *)address);
|
myArea=findArea((void *)address);
|
||||||
if (myArea)
|
if (myArea)
|
||||||
myArea->setByte(address,value);
|
myArea->setByte(address,value);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
char temp[1000];
|
||||||
|
sprintf (temp,"Unable to find an area for address %d\n",address);
|
||||||
|
throw (temp);
|
||||||
|
}
|
||||||
unlock();
|
unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -226,6 +242,12 @@ void areaManager::setInt(unsigned long address,int value)
|
|||||||
myArea=findArea((void *)address);
|
myArea=findArea((void *)address);
|
||||||
if (myArea)
|
if (myArea)
|
||||||
myArea->setInt(address,value);
|
myArea->setInt(address,value);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
char temp[1000];
|
||||||
|
sprintf (temp,"Unable to find an area for address %d\n",address);
|
||||||
|
throw (temp);
|
||||||
|
}
|
||||||
unlock();
|
unlock();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,43 +9,18 @@ class areaManager // One of these per process
|
|||||||
sem_id myLock;
|
sem_id myLock;
|
||||||
static long nextAreaID;
|
static long nextAreaID;
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
areaManager ();
|
areaManager ();
|
||||||
void addArea(area *newArea) {areas.add(newArea);}
|
void addArea(area *newArea) {areas.add(newArea);}
|
||||||
void removeArea(area *oldArea) {areas.remove(oldArea); }
|
void removeArea(area *oldArea) {areas.remove(oldArea); }
|
||||||
void freeArea(area_id area);
|
void freeArea(area_id area);
|
||||||
|
int createArea(char *AreaName,int pageCount,void **address, addressSpec addType,pageState state,protectType protect) ;
|
||||||
|
int cloneArea(int newAreaID,char *AreaName,void **address, addressSpec addType=ANY, pageState state=NO_LOCK, protectType prot=writable);
|
||||||
|
|
||||||
|
// Accessors
|
||||||
team_id getTeam(void) {return team;}
|
team_id getTeam(void) {return team;}
|
||||||
unsigned long getNextAddress(int pages,unsigned long minimum=USER_BASE);
|
unsigned long getNextAddress(int pages,unsigned long minimum=USER_BASE);
|
||||||
area *findArea(void *address);
|
status_t getAreaInfo(int areaID,area_info *dest) {
|
||||||
area *findAreaLock(void *address);
|
|
||||||
area *findArea(char *address);
|
|
||||||
area *findArea(area_id id);
|
|
||||||
area *findAreaLock(area_id id);
|
|
||||||
status_t setProtection(int areaID,protectType prot)
|
|
||||||
{
|
|
||||||
status_t retVal;
|
|
||||||
lock();
|
|
||||||
area *myArea=findArea(areaID);
|
|
||||||
if (myArea)
|
|
||||||
retVal= myArea->setProtection(prot);
|
|
||||||
else
|
|
||||||
retVal= B_ERROR;
|
|
||||||
unlock();
|
|
||||||
return retVal;
|
|
||||||
}
|
|
||||||
status_t resizeArea(int Area,size_t size)
|
|
||||||
{
|
|
||||||
status_t retVal;
|
|
||||||
lock();
|
|
||||||
area *oldArea=findArea(Area);
|
|
||||||
if (oldArea)
|
|
||||||
retVal= oldArea->resize(size);
|
|
||||||
else
|
|
||||||
retVal= B_ERROR;
|
|
||||||
unlock();
|
|
||||||
return retVal;
|
|
||||||
}
|
|
||||||
status_t getAreaInfo(int areaID,area_info *dest)
|
|
||||||
{
|
|
||||||
status_t retVal;
|
status_t retVal;
|
||||||
lock();
|
lock();
|
||||||
area *oldArea=findArea(areaID);
|
area *oldArea=findArea(areaID);
|
||||||
@ -56,8 +31,49 @@ class areaManager // One of these per process
|
|||||||
unlock();
|
unlock();
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
status_t getInfoAfter(int32 & areaID,area_info *dest)
|
int getAreaByName(char *name) {
|
||||||
{
|
int retVal;
|
||||||
|
lock();
|
||||||
|
area *oldArea=findArea(name);
|
||||||
|
if (oldArea)
|
||||||
|
retVal= oldArea->getAreaID();
|
||||||
|
else
|
||||||
|
retVal= B_ERROR;
|
||||||
|
unlock();
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
void *mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset);
|
||||||
|
status_t munmap(void *addr,size_t len);
|
||||||
|
|
||||||
|
// Mutators
|
||||||
|
area *findArea(void *address);
|
||||||
|
area *findAreaLock(void *address);
|
||||||
|
area *findArea(char *address);
|
||||||
|
area *findArea(area_id id);
|
||||||
|
area *findAreaLock(area_id id);
|
||||||
|
status_t setProtection(int areaID,protectType prot) {
|
||||||
|
status_t retVal;
|
||||||
|
lock();
|
||||||
|
area *myArea=findArea(areaID);
|
||||||
|
if (myArea)
|
||||||
|
retVal= myArea->setProtection(prot);
|
||||||
|
else
|
||||||
|
retVal= B_ERROR;
|
||||||
|
unlock();
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
status_t resizeArea(int Area,size_t size) {
|
||||||
|
status_t retVal;
|
||||||
|
lock();
|
||||||
|
area *oldArea=findArea(Area);
|
||||||
|
if (oldArea)
|
||||||
|
retVal= oldArea->resize(size);
|
||||||
|
else
|
||||||
|
retVal= B_ERROR;
|
||||||
|
unlock();
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
status_t getInfoAfter(int32 & areaID,area_info *dest) {
|
||||||
status_t retVal;
|
status_t retVal;
|
||||||
lock();
|
lock();
|
||||||
area *oldArea=findArea(areaID);
|
area *oldArea=findArea(areaID);
|
||||||
@ -72,29 +88,15 @@ class areaManager // One of these per process
|
|||||||
unlock();
|
unlock();
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
int getAreaByName(char *name)
|
|
||||||
{
|
|
||||||
int retVal;
|
|
||||||
lock();
|
|
||||||
area *oldArea=findArea(name);
|
|
||||||
if (oldArea)
|
|
||||||
retVal= oldArea->getAreaID();
|
|
||||||
else
|
|
||||||
retVal= B_ERROR;
|
|
||||||
unlock();
|
|
||||||
return retVal;
|
|
||||||
}
|
|
||||||
int createArea(char *AreaName,int pageCount,void **address, addressSpec addType,pageState state,protectType protect) ;
|
|
||||||
void pager(int desperation);
|
|
||||||
void saver(void);
|
|
||||||
void lock() { acquire_sem(myLock); }
|
void lock() { acquire_sem(myLock); }
|
||||||
void unlock() {release_sem(myLock);}
|
void unlock() {release_sem(myLock);}
|
||||||
void *mmap(void *addr, size_t len, int prot, int flags, int fd, off_t offset);
|
|
||||||
status_t munmap(void *addr,size_t len);
|
|
||||||
int cloneArea(int newAreaID,char *AreaName,void **address, addressSpec addType=ANY, pageState state=NO_LOCK, protectType prot=writable);
|
|
||||||
|
|
||||||
|
// External methods for "server" type calls
|
||||||
bool fault(void *fault_address, bool writeError); // true = OK, false = panic.
|
bool fault(void *fault_address, bool writeError); // true = OK, false = panic.
|
||||||
|
void pager(int desperation);
|
||||||
|
void saver(void);
|
||||||
|
|
||||||
|
// Debugging
|
||||||
char getByte(unsigned long offset); // This is for testing only
|
char getByte(unsigned long offset); // This is for testing only
|
||||||
void setByte(unsigned long offset,char value); // This is for testing only
|
void setByte(unsigned long offset,char value); // This is for testing only
|
||||||
int getInt(unsigned long offset); // This is for testing only
|
int getInt(unsigned long offset); // This is for testing only
|
||||||
|
@ -7,13 +7,14 @@ class poolarea
|
|||||||
list unused;
|
list unused;
|
||||||
sem_id inUse;
|
sem_id inUse;
|
||||||
public:
|
public:
|
||||||
poolarea(void)
|
// Constructors and Destructors and related
|
||||||
{
|
poolarea(void) {
|
||||||
inUse = create_sem(1,"areapool");
|
inUse = create_sem(1,"areapool");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Mutators
|
||||||
area *get(void);
|
area *get(void);
|
||||||
void put(area *in)
|
void put(area *in) {
|
||||||
{
|
|
||||||
acquire_sem(inUse);
|
acquire_sem(inUse);
|
||||||
unused.add((node *)in);
|
unused.add((node *)in);
|
||||||
release_sem(inUse);
|
release_sem(inUse);
|
||||||
|
@ -3,49 +3,49 @@
|
|||||||
#include <vpagePool.h>
|
#include <vpagePool.h>
|
||||||
#include "vmHeaderBlock.h"
|
#include "vmHeaderBlock.h"
|
||||||
|
|
||||||
bool cacheMemberIsLessThan(void *a,void *b)
|
ulong vnodeHash (node &vp) {vnode &vn=reinterpret_cast <vnode &>(vp); return vn.offset+vn.fd;}
|
||||||
{
|
bool vnodeisEqual (node &vp,node &vp2) {
|
||||||
vnode *v1 = reinterpret_cast<cacheMember *>(a)->vn;
|
vnode &vn=reinterpret_cast <vnode &>(vp);
|
||||||
vnode *v2 = reinterpret_cast<cacheMember *>(b)->vn;
|
vnode &vn2=reinterpret_cast <vnode &>(vp2);
|
||||||
if (v1->fd < v2->fd)
|
return vn.fd==vn2.fd && vn.offset==vn2.offset;
|
||||||
return true;
|
|
||||||
if ((v1->fd==v2->fd) && (v1->offset<v2->offset))
|
|
||||||
return true;
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
extern vmHeaderBlock *vmBlock;
|
extern vmHeaderBlock *vmBlock;
|
||||||
// TODO - we need to (somehow) make sure that the same vnodes here are shared with mmap.
|
// TODO - we need to (somehow) make sure that the same vnodes here are shared with mmap.
|
||||||
// Maybe a vnode manager...
|
// Maybe a vnode manager...
|
||||||
cacheManager::cacheManager(void) : area ()
|
cacheManager::cacheManager(void) : area (),cacheMembers(30) {
|
||||||
{
|
|
||||||
myLock=create_sem(1,"Cache Manager Semaphore");
|
myLock=create_sem(1,"Cache Manager Semaphore");
|
||||||
|
cacheMembers.setHash(vnodeHash);
|
||||||
|
cacheMembers.setIsEqual(vnodeisEqual);
|
||||||
}
|
}
|
||||||
|
|
||||||
void *cacheManager::findBlock(vnode *target,bool readOnly)
|
void *cacheManager::findBlock(vnode *target,bool readOnly) {
|
||||||
{
|
cacheMember *candidate=reinterpret_cast<cacheMember *>(cacheMembers.find(target));
|
||||||
if (!cacheMembers.rock)
|
|
||||||
return NULL;
|
if (!candidate || readOnly || candidate->vp->getProtection()>=writable)
|
||||||
for (struct cacheMember *cur=((cacheMember *)cacheMembers.rock);cur;cur=((cacheMember *)cur->next))
|
return candidate;
|
||||||
{
|
|
||||||
|
// At this point, we have the first one in the hahs bucket. Loop over the hash bucket from now on,
|
||||||
|
// looking for an equality and writability match...
|
||||||
|
for (struct cacheMember *cur=candidate;cur;cur=reinterpret_cast<cacheMember *>(cur->next)) {
|
||||||
if ((target==cur->vn) && (readOnly || (cur->vp->getProtection()>=writable)))
|
if ((target==cur->vn) && (readOnly || (cur->vp->getProtection()>=writable)))
|
||||||
return (cur->vp->getStartAddress());
|
return (cur->vp->getStartAddress());
|
||||||
}
|
}
|
||||||
return NULL;
|
// We didn't find one, but to get here, there has to be one that is READ ONLY. So let's make a copy
|
||||||
|
// of that one, but readable...
|
||||||
|
return createBlock(target,false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void *cacheManager::createBlock(vnode *target,bool readOnly)
|
void *cacheManager::createBlock(vnode *target,bool readOnly, cacheMember *candidate) {
|
||||||
{
|
|
||||||
bool foundSpot=false;
|
bool foundSpot=false;
|
||||||
vpage *prev=NULL,*cur=NULL;
|
vpage *prev=NULL,*cur=NULL;
|
||||||
unsigned long begin=CACHE_BEGIN;
|
unsigned long begin=CACHE_BEGIN;
|
||||||
// Find a place in the cache's virtual space to put this vnode...
|
// Find a place in the cache's virtual space to put this vnode...
|
||||||
if (vpages.rock)
|
if (vpages.rock)
|
||||||
for (cur=((vpage *)(vpages.rock));!foundSpot && cur;cur=(vpage *)(cur->next))
|
for (cur=(reinterpret_cast <vpage *>(vpages.rock));!foundSpot && cur;cur=reinterpret_cast <vpage *>(cur->next))
|
||||||
if (cur->getStartAddress()!=(void *)begin)
|
if (cur->getStartAddress()!=(void *)begin)
|
||||||
foundSpot=true;
|
foundSpot=true;
|
||||||
else // no joy
|
else { // no joy
|
||||||
{
|
|
||||||
begin+=PAGE_SIZE;
|
begin+=PAGE_SIZE;
|
||||||
prev=cur;
|
prev=cur;
|
||||||
}
|
}
|
||||||
@ -55,21 +55,23 @@ void *cacheManager::createBlock(vnode *target,bool readOnly)
|
|||||||
newPage->setup(begin,target,NULL,((readOnly)?readable:writable),NO_LOCK);
|
newPage->setup(begin,target,NULL,((readOnly)?readable:writable),NO_LOCK);
|
||||||
vpages.add(newPage);
|
vpages.add(newPage);
|
||||||
cacheMembers.add(newPage);
|
cacheMembers.add(newPage);
|
||||||
|
// While this may not seem like a good idea (since this only happens on a write),
|
||||||
|
// it is because someone may only write to part of the file/page...
|
||||||
|
if (candidate)
|
||||||
|
memcpy(newPage->getStartAddress(),candidate->vp->getStartAddress(),PAGE_SIZE);
|
||||||
unlock();
|
unlock();
|
||||||
// return address from this vnode
|
// return address from this vnode
|
||||||
return (void *)begin;
|
return (void *)begin;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void *cacheManager::readBlock(vnode *target)
|
void *cacheManager::readBlock(vnode *target) {
|
||||||
{
|
|
||||||
void *destination=findBlock(target,true);
|
void *destination=findBlock(target,true);
|
||||||
if (destination) return destination;
|
if (destination) return destination;
|
||||||
return createBlock(target,true);
|
return createBlock(target,true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void *cacheManager::writeBlock(vnode *target)
|
void *cacheManager::writeBlock(vnode *target) {
|
||||||
{
|
|
||||||
void *destination=findBlock(target,false);
|
void *destination=findBlock(target,false);
|
||||||
if (destination) return destination;
|
if (destination) return destination;
|
||||||
return createBlock(target,false);
|
return createBlock(target,false);
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
#include <vm.h>
|
#include <vm.h>
|
||||||
#include <vpage.h>
|
#include <vpage.h>
|
||||||
#include <area.h>
|
#include <area.h>
|
||||||
#include <olist.h>
|
#include <hashTable.h>
|
||||||
|
|
||||||
struct cacheMember : public node
|
struct cacheMember : public node
|
||||||
{
|
{
|
||||||
@ -13,20 +13,24 @@ struct cacheMember : public node
|
|||||||
class cacheManager : public area
|
class cacheManager : public area
|
||||||
{
|
{
|
||||||
private:
|
private:
|
||||||
orderedList cacheMembers; // Yes, this is slow and should be a hash table. This should be done prior to
|
hashTable cacheMembers;
|
||||||
// moving into the kernel, so we can test it better.
|
|
||||||
// While this very much mirrors the area's vpage list, it won't when it is a hash table...
|
|
||||||
void *findBlock (vnode *target,bool readOnly);
|
void *findBlock (vnode *target,bool readOnly);
|
||||||
void *createBlock (vnode *target,bool readOnly);
|
void *createBlock (vnode *target,bool readOnly, cacheMember *candidate=NULL);
|
||||||
sem_id myLock;
|
sem_id myLock;
|
||||||
public:
|
public:
|
||||||
// For these two, the VFS passes in the target vnode
|
// For these two, the VFS passes in the target vnode
|
||||||
// Return value is the address. Note that the paging daemon does the actual loading
|
// Return value is the address. Note that the paging daemon does the actual loading
|
||||||
|
|
||||||
|
// Constructors and Destructors and related
|
||||||
cacheManager(void);
|
cacheManager(void);
|
||||||
|
|
||||||
|
// Mutators
|
||||||
void *readBlock (vnode *target);
|
void *readBlock (vnode *target);
|
||||||
void *writeBlock (vnode *target);
|
void *writeBlock (vnode *target);
|
||||||
void pager(int desperation); // override, as we should blow away useless nodes, not just free blocks.
|
|
||||||
void saver(void); // Override - not sure why
|
|
||||||
void lock() {acquire_sem(myLock);}
|
void lock() {acquire_sem(myLock);}
|
||||||
void unlock() {release_sem(myLock);}
|
void unlock() {release_sem(myLock);}
|
||||||
|
|
||||||
|
// External methods for "server" type calls
|
||||||
|
void pager(int desperation); // override, as we should blow away useless nodes, not just free blocks.
|
||||||
|
void saver(void); // Override - not sure why
|
||||||
};
|
};
|
||||||
|
@ -10,7 +10,7 @@ void error(char *fmt, ...)
|
|||||||
errorPrinting=create_sem(1,"error_printing");
|
errorPrinting=create_sem(1,"error_printing");
|
||||||
acquire_sem(errorPrinting);
|
acquire_sem(errorPrinting);
|
||||||
va_list argp;
|
va_list argp;
|
||||||
char tmp[1000];
|
char tmp[2000];
|
||||||
sprintf(tmp, "[%lld] error: %s",real_time_clock_usecs(),fmt);
|
sprintf(tmp, "[%lld] error: %s",real_time_clock_usecs(),fmt);
|
||||||
va_start(argp, tmp);
|
va_start(argp, tmp);
|
||||||
vfprintf(stderr,tmp,argp);
|
vfprintf(stderr,tmp,argp);
|
||||||
|
@ -9,19 +9,27 @@
|
|||||||
|
|
||||||
extern vmHeaderBlock *vmBlock;
|
extern vmHeaderBlock *vmBlock;
|
||||||
|
|
||||||
|
class hashIterate;
|
||||||
|
|
||||||
class hashTable : public list
|
class hashTable : public list
|
||||||
{
|
{
|
||||||
|
friend hashIterate;
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
hashTable(int size) {
|
hashTable(int size) {
|
||||||
nodeCount=0;
|
nodeCount=0;
|
||||||
numRocks=size;
|
numRocks=size;
|
||||||
|
|
||||||
|
error ("Starting to initalize hash table\n");
|
||||||
if (size*sizeof (list *)>PAGE_SIZE)
|
if (size*sizeof (list *)>PAGE_SIZE)
|
||||||
throw ("Hash table too big!");
|
throw ("Hash table too big!");
|
||||||
|
error ("Getting Page\n");
|
||||||
page *newPage=vmBlock->pageMan->getPage();
|
page *newPage=vmBlock->pageMan->getPage();
|
||||||
|
error ("Got Page\n");
|
||||||
if (!newPage)
|
if (!newPage)
|
||||||
throw ("Out of pages to allocate a pool!");
|
throw ("Out of pages to allocate a pool!");
|
||||||
rocks=(list **)(newPage->getAddress());
|
rocks=(list **)(newPage->getAddress());
|
||||||
|
error ("Got rocks\n");
|
||||||
|
|
||||||
int listsPerPage=PAGE_SIZE/sizeof(list);
|
int listsPerPage=PAGE_SIZE/sizeof(list);
|
||||||
int pages=(size+(listsPerPage-1))/listsPerPage;
|
int pages=(size+(listsPerPage-1))/listsPerPage;
|
||||||
@ -35,10 +43,9 @@ class hashTable : public list
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Mutators
|
||||||
void setHash (ulong (*hash_in)(node &)) { hash=hash_in; }
|
void setHash (ulong (*hash_in)(node &)) { hash=hash_in; }
|
||||||
void setIsEqual (bool (*isEqual_in)(node &,node &)) { isEqual=isEqual_in; }
|
void setIsEqual (bool (*isEqual_in)(node &,node &)) { isEqual=isEqual_in; }
|
||||||
|
|
||||||
int count(void) {return nodeCount;}
|
|
||||||
void add (node *newNode) {
|
void add (node *newNode) {
|
||||||
if (!hash)
|
if (!hash)
|
||||||
throw ("Attempting to use a hash table without setting up a 'hash' function");
|
throw ("Attempting to use a hash table without setting up a 'hash' function");
|
||||||
@ -47,27 +54,8 @@ class hashTable : public list
|
|||||||
rocks[hashValue]->add(newNode);
|
rocks[hashValue]->add(newNode);
|
||||||
}
|
}
|
||||||
|
|
||||||
node *next(void) {throw ("Next is invalid in a hash table!");} // This operation doesn't make sense for this class
|
// Accessors
|
||||||
|
int count(void) {return nodeCount;}
|
||||||
void remove(node *toNuke) {
|
|
||||||
if (!hash)
|
|
||||||
throw ("Attempting to use a hash table without setting up a 'hash' function");
|
|
||||||
unsigned long hashValue=hash(*toNuke)%numRocks;
|
|
||||||
rocks[hashValue]->remove(toNuke);
|
|
||||||
}
|
|
||||||
|
|
||||||
void dump(void) {
|
|
||||||
for (int i=0;i<numRocks;i++)
|
|
||||||
for (struct node *cur=rocks[i]->rock;cur;cur=cur->next)
|
|
||||||
error ("hashTable::dump: On bucket %d of %d, At %p, next = %p\n",i,numRocks,cur,cur->next);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ensureSane (void) {
|
|
||||||
bool ok=true;
|
|
||||||
for (int i=0;i<numRocks;i++)
|
|
||||||
ok|=rocks[i]->ensureSane();
|
|
||||||
return ok;
|
|
||||||
}
|
|
||||||
node *find(node *findNode) {
|
node *find(node *findNode) {
|
||||||
if (!hash)
|
if (!hash)
|
||||||
throw ("Attempting to use a hash table without setting up a 'hash' function");
|
throw ("Attempting to use a hash table without setting up a 'hash' function");
|
||||||
@ -79,6 +67,26 @@ class hashTable : public list
|
|||||||
return cur;
|
return cur;
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
node *next(void) {throw ("Next is invalid in a hash table!");} // This operation doesn't make sense for this class
|
||||||
|
void remove(node *toNuke) {
|
||||||
|
if (!hash)
|
||||||
|
throw ("Attempting to use a hash table without setting up a 'hash' function");
|
||||||
|
unsigned long hashValue=hash(*toNuke)%numRocks;
|
||||||
|
rocks[hashValue]->remove(toNuke);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Debugging
|
||||||
|
void dump(void) {
|
||||||
|
for (int i=0;i<numRocks;i++)
|
||||||
|
for (struct node *cur=rocks[i]->rock;cur;cur=cur->next)
|
||||||
|
error ("hashTable::dump: On bucket %d of %d, At %p, next = %p\n",i,numRocks,cur,cur->next);
|
||||||
|
}
|
||||||
|
bool ensureSane (void) {
|
||||||
|
bool ok=true;
|
||||||
|
for (int i=0;i<numRocks;i++)
|
||||||
|
ok|=rocks[i]->ensureSane();
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ulong (*hash)(node &a);
|
ulong (*hash)(node &a);
|
||||||
@ -86,4 +94,27 @@ class hashTable : public list
|
|||||||
list **rocks;
|
list **rocks;
|
||||||
int numRocks;
|
int numRocks;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class hashIterate {
|
||||||
|
private:
|
||||||
|
int bucket;
|
||||||
|
node *current;
|
||||||
|
hashTable &table;
|
||||||
|
|
||||||
|
public:
|
||||||
|
hashIterate(hashTable *in) : bucket(0),current(NULL),table(*in) {}
|
||||||
|
hashIterate(hashTable &in) : bucket(0),current(NULL),table(in) {}
|
||||||
|
|
||||||
|
node *get(void)
|
||||||
|
{
|
||||||
|
while (!current && bucket<table.numRocks)
|
||||||
|
current=table.rocks[bucket++]->top();
|
||||||
|
if (!current)
|
||||||
|
return NULL; // No more in the hash table
|
||||||
|
node *retVal=current; // Store the current and get then next in preperation
|
||||||
|
current=current->next;
|
||||||
|
return retVal;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -27,7 +27,7 @@ bool isEqual (node &a,node &b) {
|
|||||||
int main(int argc,char **argv) {
|
int main(int argc,char **argv) {
|
||||||
// Test 1 - Try to add to foo without setting an isLessThan function
|
// Test 1 - Try to add to foo without setting an isLessThan function
|
||||||
try {
|
try {
|
||||||
hashTable foo(10);
|
hashTable foo(40);
|
||||||
node tmp;
|
node tmp;
|
||||||
foo.add(&tmp);
|
foo.add(&tmp);
|
||||||
}
|
}
|
||||||
@ -68,7 +68,7 @@ int main(int argc,char **argv) {
|
|||||||
bar.setIsEqual(isEqual);
|
bar.setIsEqual(isEqual);
|
||||||
for (int a=0;a<100;a++)
|
for (int a=0;a<100;a++)
|
||||||
bar.add(new hashTest(a));
|
bar.add(new hashTest(a));
|
||||||
for (int a=1000;a>=100;a--)
|
for (int a=999;a>=100;a--)
|
||||||
bar.add(new hashTest(a));
|
bar.add(new hashTest(a));
|
||||||
bar.dump();
|
bar.dump();
|
||||||
|
|
||||||
@ -79,7 +79,7 @@ int main(int argc,char **argv) {
|
|||||||
foo.remove(&second);
|
foo.remove(&second);
|
||||||
foo.dump();
|
foo.dump();
|
||||||
|
|
||||||
error ("Setting up middle case (readding 1 and 2)\n");
|
error ("Setting up middle case (reading 1 and 2)\n");
|
||||||
foo.add(&first);
|
foo.add(&first);
|
||||||
foo.add(&second);
|
foo.add(&second);
|
||||||
foo.dump();
|
foo.dump();
|
||||||
@ -103,5 +103,16 @@ int main(int argc,char **argv) {
|
|||||||
error ("ensuringSane on smaller (foo) = %s\n",((foo.ensureSane())?"OK":"BAD!"));
|
error ("ensuringSane on smaller (foo) = %s\n",((foo.ensureSane())?"OK":"BAD!"));
|
||||||
error ("ensuringSane on larger (bar) = %s\n",((bar.ensureSane())?"OK":"BAD!"));
|
error ("ensuringSane on larger (bar) = %s\n",((bar.ensureSane())?"OK":"BAD!"));
|
||||||
|
|
||||||
|
|
||||||
|
int count=0;
|
||||||
|
for (hashIterate hi(bar);node *next=hi.get();) {
|
||||||
|
count++;
|
||||||
|
if (next==NULL)
|
||||||
|
error ("Found a NULL at %d\n",count);
|
||||||
|
}
|
||||||
|
if (count==1000)
|
||||||
|
error ("found 1000, as expected!\n");
|
||||||
|
else
|
||||||
|
error ("did NOT find 1000, as expected, found %d!\n",count);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -12,14 +12,15 @@ struct node
|
|||||||
|
|
||||||
class list {
|
class list {
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
list(void){nodeCount=0;rock=NULL;}
|
list(void){nodeCount=0;rock=NULL;}
|
||||||
|
|
||||||
|
// Mutators
|
||||||
void add (node *newNode) {
|
void add (node *newNode) {
|
||||||
newNode->next=rock;
|
newNode->next=rock;
|
||||||
rock=newNode;
|
rock=newNode;
|
||||||
nodeCount++;
|
nodeCount++;
|
||||||
}
|
}
|
||||||
//int count(void) {error ("list::count: About to return %d\n",nodeCount);return nodeCount;}
|
|
||||||
int count(void) {return nodeCount;}
|
|
||||||
node *next(void) {
|
node *next(void) {
|
||||||
//dump();
|
//dump();
|
||||||
node *n=rock;
|
node *n=rock;
|
||||||
@ -49,11 +50,16 @@ class list {
|
|||||||
//error ("list::remove ending: \n");
|
//error ("list::remove ending: \n");
|
||||||
//dump();
|
//dump();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Accessors
|
||||||
|
int count(void) {return nodeCount;}
|
||||||
|
node *top(void) {return rock;} // Intentionally non-destructive ; works like peek() on a queue
|
||||||
|
|
||||||
|
// Debugging
|
||||||
void dump(void) {
|
void dump(void) {
|
||||||
for (struct node *cur=rock;cur;cur=cur->next)
|
for (struct node *cur=rock;cur;cur=cur->next)
|
||||||
{ error ("list::dump: At %p, next = %p\n",cur,cur->next); }
|
{ error ("list::dump: At %p, next = %p\n",cur,cur->next); }
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ensureSane (void) {
|
bool ensureSane (void) {
|
||||||
int temp=nodeCount;
|
int temp=nodeCount;
|
||||||
for (struct node *cur=rock;cur && --temp;cur=cur->next) ; // Intentional to have no body
|
for (struct node *cur=rock;cur && --temp;cur=cur->next) ; // Intentional to have no body
|
||||||
|
@ -10,27 +10,24 @@ static bool throwException (void *foo, void *bar)
|
|||||||
class orderedList : public list
|
class orderedList : public list
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
orderedList(void) {nodeCount=0;rock=NULL; isLessThan=throwException; }
|
orderedList(void) {nodeCount=0;rock=NULL; isLessThan=throwException; }
|
||||||
|
|
||||||
|
// Mutators
|
||||||
void setIsLessThan (bool (*iLT)(void *,void *)) { isLessThan=iLT; }
|
void setIsLessThan (bool (*iLT)(void *,void *)) { isLessThan=iLT; }
|
||||||
|
void add(node *in) {
|
||||||
void add(node *in)
|
|
||||||
{
|
|
||||||
nodeCount++;
|
nodeCount++;
|
||||||
//error ("orderedList::add starting\n");
|
//error ("orderedList::add starting\n");
|
||||||
if (!rock || isLessThan(in,rock))
|
if (!rock || isLessThan(in,rock)) { // special case - this will be the first one
|
||||||
{ // special case - this will be the first one
|
|
||||||
//error ("orderedList::specialCase starting\n");
|
//error ("orderedList::specialCase starting\n");
|
||||||
in->next=rock;
|
in->next=rock;
|
||||||
rock=in;
|
rock=in;
|
||||||
}
|
}
|
||||||
else
|
else {
|
||||||
{
|
|
||||||
//error ("orderedList::Normal Case starting\n");
|
//error ("orderedList::Normal Case starting\n");
|
||||||
bool done=false;
|
bool done=false;
|
||||||
for (struct node *cur=rock;cur && !done;cur=cur->next)
|
for (struct node *cur=rock;cur && !done;cur=cur->next)
|
||||||
if (!(cur->next) || isLessThan(in,cur->next))
|
if (!(cur->next) || isLessThan(in,cur->next)) { // If we have found our niche, *OR* this is the last element, insert here.
|
||||||
{ // If we have found our niche, *OR* this is the last element, insert here.
|
|
||||||
//error ("orderedList::Normal Case Adding Start\n");
|
//error ("orderedList::Normal Case Adding Start\n");
|
||||||
in->next=cur->next;
|
in->next=cur->next;
|
||||||
cur->next=in;
|
cur->next=in;
|
||||||
@ -40,20 +37,15 @@ class orderedList : public list
|
|||||||
//error ("orderedList::Normal Case ending\n");
|
//error ("orderedList::Normal Case ending\n");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
void remove(node *toNuke) {
|
||||||
void remove(node *toNuke)
|
if (rock==toNuke) {
|
||||||
{
|
|
||||||
if (rock==toNuke)
|
|
||||||
{
|
|
||||||
rock=rock->next;
|
rock=rock->next;
|
||||||
nodeCount--;
|
nodeCount--;
|
||||||
}
|
}
|
||||||
else
|
else {
|
||||||
{
|
|
||||||
bool done=false;
|
bool done=false;
|
||||||
for (struct node *cur=rock;!done && (cur->next);cur=cur->next)
|
for (struct node *cur=rock;!done && (cur->next);cur=cur->next)
|
||||||
if (cur->next==toNuke)
|
if (cur->next==toNuke) {
|
||||||
{
|
|
||||||
cur->next=toNuke->next;
|
cur->next=toNuke->next;
|
||||||
nodeCount--;
|
nodeCount--;
|
||||||
done=true;
|
done=true;
|
||||||
|
@ -7,11 +7,19 @@ class page : public node {
|
|||||||
void *cpuSpecific;
|
void *cpuSpecific;
|
||||||
void *physicalAddress;
|
void *physicalAddress;
|
||||||
public:
|
public:
|
||||||
long count; // Yes, this is large. However, the only atomic add that I have in userland works on int32's. In kernel land, we could shrink this
|
// Constructors and Destructors and related
|
||||||
page(void) {cpuSpecific=NULL;physicalAddress=NULL;};
|
page(void) {cpuSpecific=NULL;physicalAddress=NULL;};
|
||||||
void setup (void *address) {count=0;physicalAddress=address;};
|
void setup (void *address) {count=0;physicalAddress=address;};
|
||||||
void zero(void);
|
|
||||||
|
// Accessors
|
||||||
unsigned long getAddress(void) {return (unsigned long)physicalAddress;}
|
unsigned long getAddress(void) {return (unsigned long)physicalAddress;}
|
||||||
|
|
||||||
|
// Debugging
|
||||||
void dump(void) { error ("page::dump: Page %p, physicalAddress = %lx\n",this,getAddress()); }
|
void dump(void) { error ("page::dump: Page %p, physicalAddress = %lx\n",this,getAddress()); }
|
||||||
|
|
||||||
|
// External methods for "server" type calls
|
||||||
|
void zero(void);
|
||||||
|
|
||||||
|
long count; // Yes, this is large. However, the only atomic add that I have in userland works on int32's. In kernel land, we could shrink this
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
102
src/kernel/vm2/pageManTest.C
Normal file
102
src/kernel/vm2/pageManTest.C
Normal file
@ -0,0 +1,102 @@
|
|||||||
|
#include <stdio.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include "pageManager.h"
|
||||||
|
#include "OS.h"
|
||||||
|
|
||||||
|
enum testType {clean,unused,inuse};
|
||||||
|
|
||||||
|
pageManager pm;
|
||||||
|
char *testName;
|
||||||
|
|
||||||
|
bool testCount (int expected, testType type)
|
||||||
|
{
|
||||||
|
int found;
|
||||||
|
switch (type) {
|
||||||
|
case clean:
|
||||||
|
if (found=pm.getCleanCount()!=expected) {
|
||||||
|
error ("%s, invalid cleanCount of %d, expected: %d\n",testName,found,expected);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case unused:
|
||||||
|
if (found=pm.getUnusedCount()!=expected) {
|
||||||
|
error ("%s, invalid unusedCount of %d, expected: %d\n",testName,found,expected);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case inuse:
|
||||||
|
if (found=pm.getInUseCount()!=expected) {
|
||||||
|
error ("%s, invalid InUseCount of %d, expected: %d\n",testName,found,expected);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc,char **argv)
|
||||||
|
{
|
||||||
|
testName="Setup";
|
||||||
|
|
||||||
|
pm.setup(malloc(PAGE_SIZE*100),100);
|
||||||
|
|
||||||
|
if (!(testCount(99,unused) || testCount (0,clean) || testCount (0,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Get One";
|
||||||
|
page *gotten = pm.getPage();
|
||||||
|
if (!(testCount(98,unused) || testCount (0,clean) || testCount (1,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Put One";
|
||||||
|
pm.freePage(gotten);
|
||||||
|
if (!(testCount(99,unused) || testCount (0,clean) || testCount (0,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Clean One";
|
||||||
|
pm.cleanOnePage();
|
||||||
|
if (!(testCount(98,unused) || testCount (1,clean) || testCount (0,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Get One Cleaned";
|
||||||
|
gotten = pm.getPage();
|
||||||
|
if (!(testCount(98,unused) || testCount (0,clean) || testCount (1,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Put One Cleaned";
|
||||||
|
pm.freePage(gotten);
|
||||||
|
if (!(testCount(99,unused) || testCount (0,clean) || testCount (0,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Get them all";
|
||||||
|
page *pages[100];
|
||||||
|
for (int i=0;i<99;i++)
|
||||||
|
pages[i]=pm.getPage();
|
||||||
|
if (!(testCount(0,unused) || testCount (0,clean) || testCount (99,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
testName="Free them all";
|
||||||
|
for (int i=0;i<99;i++)
|
||||||
|
pm.freePage(pages[i]);
|
||||||
|
if (!(testCount(99,unused) || testCount (0,clean) || testCount (0,inuse))) {
|
||||||
|
pm.dump();
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
exit(0);
|
||||||
|
}
|
@ -3,26 +3,19 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
void *addOffset(void *base,unsigned long offset)
|
void *addOffset(void *base,unsigned long offset) {
|
||||||
{
|
|
||||||
return (void *)(((unsigned long)base+offset));
|
return (void *)(((unsigned long)base+offset));
|
||||||
}
|
}
|
||||||
|
|
||||||
pageManager::pageManager(void)
|
pageManager::pageManager(void) {
|
||||||
{
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void pageManager::setup(void *area,int pages)
|
void pageManager::setup(void *area,int pages) {
|
||||||
{
|
|
||||||
// Calculate the number of pages that we will need to hold the page structures
|
// Calculate the number of pages that we will need to hold the page structures
|
||||||
int pageOverhead=((pages*sizeof(page))+(PAGE_SIZE-1))/PAGE_SIZE;
|
int pageOverhead=((pages*sizeof(page))+(PAGE_SIZE-1))/PAGE_SIZE;
|
||||||
for (int i=0;i<pages-pageOverhead;i++)
|
for (int i=0;i<pages-pageOverhead;i++) {
|
||||||
{
|
|
||||||
page *newPage=(page *)(addOffset(area,i*sizeof(page)));
|
page *newPage=(page *)(addOffset(area,i*sizeof(page)));
|
||||||
newPage->setup(addOffset(area,(i+pageOverhead)*PAGE_SIZE));
|
newPage->setup(addOffset(area,(i+pageOverhead)*PAGE_SIZE));
|
||||||
//error ("newPage = %x, setup = %x\n",newPage,addOffset(area,(i+pageOverhead)*PAGE_SIZE));
|
|
||||||
//error ("i = %d, sizeof(page) = %x, pageOverhead = %d\n",i,sizeof(page),pageOverhead);
|
|
||||||
//newPage->dump();
|
|
||||||
unused.add(newPage);
|
unused.add(newPage);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -30,39 +23,21 @@ void pageManager::setup(void *area,int pages)
|
|||||||
unusedLock=create_sem (1,"unused_lock");
|
unusedLock=create_sem (1,"unused_lock");
|
||||||
inUseLock=create_sem (1,"inuse_lock");
|
inUseLock=create_sem (1,"inuse_lock");
|
||||||
totalPages=pages;
|
totalPages=pages;
|
||||||
/*
|
|
||||||
error ("pageManager::pageManager: About to dump the clean pages (should be 0):\n\n");
|
|
||||||
clean.dump();
|
|
||||||
error ("pageManager::pageManager: About to dump the unused pages (should not be 0):\n\n");
|
|
||||||
unused.dump();
|
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
|
|
||||||
page *pageManager::getPage(void)
|
page *pageManager::getPage(void) {
|
||||||
{
|
|
||||||
page *ret=NULL;
|
page *ret=NULL;
|
||||||
// error ("pageManager::getPage: Checking clean\n");
|
|
||||||
//error ("pageManager::getPage:cleanCount = %d\n", clean.nodeCount);
|
|
||||||
while (!ret)
|
while (!ret)
|
||||||
{
|
{
|
||||||
if (clean.count())
|
if (clean.count()) {
|
||||||
{
|
|
||||||
//error ("pageManager::getPage:locking clean\n");
|
|
||||||
acquire_sem(cleanLock);
|
acquire_sem(cleanLock);
|
||||||
//error ("pageManager::getPage:locked clean\n");
|
|
||||||
ret=(page *)clean.next();
|
ret=(page *)clean.next();
|
||||||
//error ("pageManager::getPage:got next clean\n");
|
|
||||||
release_sem(cleanLock);
|
release_sem(cleanLock);
|
||||||
//error ("pageManager::getPage:unlocked clean\n");
|
|
||||||
} // This could fail if someone swooped in and stole our page.
|
} // This could fail if someone swooped in and stole our page.
|
||||||
if (!ret && unused.count())
|
if (!ret && unused.count()) {
|
||||||
{
|
|
||||||
//error ("pageManager::getPage:Checking unused\n");
|
|
||||||
acquire_sem(unusedLock);
|
acquire_sem(unusedLock);
|
||||||
ret=(page *)unused.next();
|
ret=(page *)unused.next();
|
||||||
//error ("pageManager::getPage:got next unused\n");
|
|
||||||
release_sem(unusedLock);
|
release_sem(unusedLock);
|
||||||
//error ("pageManager::getPage:next unused = %x\n",ret);
|
|
||||||
if (ret)
|
if (ret)
|
||||||
ret->zero();
|
ret->zero();
|
||||||
} // This could fail if someone swooped in and stole our page.
|
} // This could fail if someone swooped in and stole our page.
|
||||||
@ -71,37 +46,30 @@ page *pageManager::getPage(void)
|
|||||||
inUse.add(ret);
|
inUse.add(ret);
|
||||||
release_sem(inUseLock);
|
release_sem(inUseLock);
|
||||||
ret->count++;
|
ret->count++;
|
||||||
// error ("pageManager::getPage:leaving with page = %x\n", ret->getAddress());
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool pageManager::getContiguousPages(int pages,page **location)
|
bool pageManager::getContiguousPages(int pages,page **location) {
|
||||||
{
|
|
||||||
unsigned long current, start=0, next;
|
unsigned long current, start=0, next;
|
||||||
page *curPage;
|
page *curPage;
|
||||||
int count=0;
|
int count=0;
|
||||||
while (count<pages)
|
while (count<pages) {
|
||||||
{
|
|
||||||
curPage=getPage();
|
curPage=getPage();
|
||||||
current=curPage->getAddress();
|
current=curPage->getAddress();
|
||||||
if (start==0)
|
if (start==0) {
|
||||||
{
|
|
||||||
start=current;
|
start=current;
|
||||||
location[count++]=curPage;
|
location[count++]=curPage;
|
||||||
}
|
}
|
||||||
else if (current==start+PAGE_SIZE*count) // This is the next one in line
|
else if (current==start+PAGE_SIZE*count) // This is the next one in line
|
||||||
location[count++]=curPage;
|
location[count++]=curPage;
|
||||||
else if (current==start-PAGE_SIZE) // Found the one directly previous
|
else if (current==start-PAGE_SIZE) { // Found the one directly previous
|
||||||
{
|
|
||||||
memmove(location[1],location[0],count*sizeof(page *));
|
memmove(location[1],location[0],count*sizeof(page *));
|
||||||
start=current;
|
start=current;
|
||||||
location[0]=curPage;
|
location[0]=curPage;
|
||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
else // Forget this series - it doesn't seem to be going anywhere...
|
else { // Forget this series - it doesn't seem to be going anywhere...
|
||||||
{
|
while (--count>=0) {
|
||||||
while (--count>=0)
|
|
||||||
{
|
|
||||||
freePage(location[count]);
|
freePage(location[count]);
|
||||||
location[count]=NULL;
|
location[count]=NULL;
|
||||||
}
|
}
|
||||||
@ -110,11 +78,8 @@ bool pageManager::getContiguousPages(int pages,page **location)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void pageManager::freePage(page *toFree)
|
void pageManager::freePage(page *toFree) {
|
||||||
{
|
if (atomic_add(&(toFree->count),-1)==1) { // atomic_add returns the *PREVIOUS* value. So we need to check to see if the one we are wasting was the last one.
|
||||||
error ("pageManager::freePage; old value = %d\n",toFree->count);
|
|
||||||
if (atomic_add(&(toFree->count),-1)==1) // atomic_add returns the *PREVIOUS* value. So we need to check to see if the one we are wasting was the last one.
|
|
||||||
{
|
|
||||||
acquire_sem(inUseLock);
|
acquire_sem(inUseLock);
|
||||||
inUse.remove(toFree);
|
inUse.remove(toFree);
|
||||||
release_sem(inUseLock);
|
release_sem(inUseLock);
|
||||||
@ -122,63 +87,55 @@ void pageManager::freePage(page *toFree)
|
|||||||
unused.add(toFree);
|
unused.add(toFree);
|
||||||
release_sem(unusedLock);
|
release_sem(unusedLock);
|
||||||
}
|
}
|
||||||
error ("pageManager::freePage:new value = %d, page = %x\n",toFree->count,toFree->getAddress());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void pageManager::cleaner(void)
|
void pageManager::cleaner(void) {
|
||||||
{
|
while (1) {
|
||||||
while (1)
|
|
||||||
{
|
|
||||||
snooze(250000);
|
snooze(250000);
|
||||||
if (unused.count())
|
cleanOnePage();
|
||||||
{
|
}
|
||||||
//error ("pageManager::cleaner: About to vacuum a page\n");
|
}
|
||||||
|
|
||||||
|
void pageManager::cleanOnePage(void) {
|
||||||
|
if (unused.count()) {
|
||||||
acquire_sem(unusedLock);
|
acquire_sem(unusedLock);
|
||||||
page *first=(page *)unused.next();
|
page *first=(page *)unused.next();
|
||||||
release_sem(unusedLock);
|
release_sem(unusedLock);
|
||||||
if (first)
|
if (first) {
|
||||||
{
|
|
||||||
first->zero();
|
first->zero();
|
||||||
acquire_sem(cleanLock);
|
acquire_sem(cleanLock);
|
||||||
clean.add(first);
|
clean.add(first);
|
||||||
release_sem(cleanLock);
|
release_sem(cleanLock);
|
||||||
}
|
}
|
||||||
//error ("pageManager::cleaner: All done with vacuum a page\n");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int pageManager::desperation(void)
|
int pageManager::desperation(void) { // Formula to determine how desperate system is to get pages back...
|
||||||
{ // Formula to determine how desperate system is to get pages back...
|
|
||||||
int percentClean=(unused.count()+clean.count())*100/totalPages;
|
int percentClean=(unused.count()+clean.count())*100/totalPages;
|
||||||
if (percentClean>30) return 1;
|
if (percentClean>30) return 1;
|
||||||
return (35-percentClean)/7;
|
return (35-percentClean)/7;
|
||||||
}
|
}
|
||||||
|
|
||||||
void pageManager::dump(void)
|
void pageManager::dump(void) {
|
||||||
{
|
error ("Dumping the unused list (%d entries)\n",getUnusedCount());
|
||||||
error ("Dumping the unused list\n");
|
|
||||||
acquire_sem(unusedLock);
|
acquire_sem(unusedLock);
|
||||||
for (struct node *cur=unused.rock;cur;)
|
for (struct node *cur=unused.rock;cur;) {
|
||||||
{
|
|
||||||
page *thisPage=(page *)cur;
|
page *thisPage=(page *)cur;
|
||||||
thisPage->dump();
|
thisPage->dump();
|
||||||
cur=cur->next;
|
cur=cur->next;
|
||||||
}
|
}
|
||||||
release_sem(unusedLock);
|
release_sem(unusedLock);
|
||||||
error ("Dumping the clean list\n");
|
error ("Dumping the clean list (%d entries)\n",getCleanCount());
|
||||||
acquire_sem(cleanLock);
|
acquire_sem(cleanLock);
|
||||||
for (struct node *cur=clean.rock;cur;)
|
for (struct node *cur=clean.rock;cur;) {
|
||||||
{
|
|
||||||
page *thisPage=(page *)cur;
|
page *thisPage=(page *)cur;
|
||||||
thisPage->dump();
|
thisPage->dump();
|
||||||
cur=cur->next;
|
cur=cur->next;
|
||||||
}
|
}
|
||||||
error ("Dumping the inuse list\n");
|
error ("Dumping the inuse list (%d entries)\n",getInUseCount());
|
||||||
release_sem(cleanLock);
|
release_sem(cleanLock);
|
||||||
acquire_sem(inUseLock);
|
acquire_sem(inUseLock);
|
||||||
for (struct node *cur=inUse.rock;cur;)
|
for (struct node *cur=inUse.rock;cur;) {
|
||||||
{
|
|
||||||
page *thisPage=(page *)cur;
|
page *thisPage=(page *)cur;
|
||||||
thisPage->dump();
|
thisPage->dump();
|
||||||
cur=cur->next;
|
cur=cur->next;
|
||||||
|
@ -6,14 +6,27 @@
|
|||||||
|
|
||||||
class pageManager {
|
class pageManager {
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
pageManager(void);
|
pageManager(void);
|
||||||
void setup(void *memory,int pages);
|
void setup(void *memory,int pages);
|
||||||
|
void freePage(page *);
|
||||||
|
|
||||||
|
// Mutators
|
||||||
page *getPage(void);
|
page *getPage(void);
|
||||||
bool getContiguousPages(int pages,page **location);
|
bool getContiguousPages(int pages,page **location);
|
||||||
void freePage(page *);
|
|
||||||
void cleaner(void);
|
// Accessors
|
||||||
int desperation(void);
|
int desperation(void);
|
||||||
|
|
||||||
|
// External methods for "server" type calls
|
||||||
|
void cleaner(void);
|
||||||
|
void cleanOnePage(void);
|
||||||
|
|
||||||
|
// Debugging
|
||||||
void dump(void);
|
void dump(void);
|
||||||
|
int getCleanCount(void) {return clean.count();}
|
||||||
|
int getUnusedCount(void) {return unused.count();}
|
||||||
|
int getInUseCount(void) {return inUse.count();}
|
||||||
private:
|
private:
|
||||||
list clean,unused,inUse;
|
list clean,unused,inUse;
|
||||||
sem_id cleanLock,unusedLock,inUseLock;
|
sem_id cleanLock,unusedLock,inUseLock;
|
||||||
|
@ -12,27 +12,21 @@ class poolTYPE
|
|||||||
list unused;
|
list unused;
|
||||||
sem_id inUse;
|
sem_id inUse;
|
||||||
public:
|
public:
|
||||||
poolTYPE(void)
|
poolTYPE(void) { inUse = create_sem(1,"TYPEpool");
|
||||||
{
|
|
||||||
inUse = create_sem(1,"TYPEpool");
|
|
||||||
}
|
}
|
||||||
TYPE *get(void)
|
TYPE *get(void) {
|
||||||
{
|
|
||||||
TYPE *ret=NULL;
|
TYPE *ret=NULL;
|
||||||
if (unused.count())
|
if (unused.count()) {
|
||||||
{
|
|
||||||
error ("poolTYPE::get: Getting an unused one!\n");
|
error ("poolTYPE::get: Getting an unused one!\n");
|
||||||
acquire_sem(inUse);
|
acquire_sem(inUse);
|
||||||
ret=(TYPE *)unused.next();
|
ret=(TYPE *)unused.next();
|
||||||
release_sem(inUse);
|
release_sem(inUse);
|
||||||
}
|
}
|
||||||
if (ret)
|
if (ret) {
|
||||||
{
|
|
||||||
error ("poolTYPE::get: Returning address:%x \n",ret);
|
error ("poolTYPE::get: Returning address:%x \n",ret);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
else
|
else {
|
||||||
{
|
|
||||||
error ("poolTYPE::get: Getting a new page!\n");
|
error ("poolTYPE::get: Getting a new page!\n");
|
||||||
page *newPage=vmBlock->pageMan->getPage();
|
page *newPage=vmBlock->pageMan->getPage();
|
||||||
if (!newPage)
|
if (!newPage)
|
||||||
@ -46,12 +40,11 @@ class poolTYPE
|
|||||||
return (get()); // A little cheat - call self again to get the first one from stack...
|
return (get()); // A little cheat - call self again to get the first one from stack...
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void put(TYPE *in)
|
|
||||||
{
|
void put(TYPE *in) {
|
||||||
acquire_sem(inUse);
|
acquire_sem(inUse);
|
||||||
unused.add(in);
|
unused.add(in);
|
||||||
release_sem(inUse);
|
release_sem(inUse);
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
*/
|
*/
|
||||||
|
57
src/kernel/vm2/simpleTest.C
Normal file
57
src/kernel/vm2/simpleTest.C
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
#include "vmInterface.h"
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
#include <mman.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
vmInterface vm(30);
|
||||||
|
|
||||||
|
void writeByte(unsigned long addr,unsigned int offset, char value) { vm.setByte(addr+offset,value); }
|
||||||
|
unsigned char readByte(unsigned long addr,unsigned int offset ) { char value=vm.getByte(addr+offset); return value; }
|
||||||
|
|
||||||
|
int createFillAndTest(int pages,char *name)
|
||||||
|
{
|
||||||
|
try{
|
||||||
|
unsigned long addr;
|
||||||
|
int area1;
|
||||||
|
error ("%s: createFillAndTest: about to create \n",name);
|
||||||
|
area1=vm.createArea(name,pages,(void **)(&addr));
|
||||||
|
error ("%s: createFillAndTest: create done\n",name);
|
||||||
|
for (int i=0;i<pages*PAGE_SIZE;i++)
|
||||||
|
{
|
||||||
|
if (!(i%9024) )
|
||||||
|
error ("Writing to byte %d of area %s\n",i,name);
|
||||||
|
writeByte(addr,i,i%256);
|
||||||
|
}
|
||||||
|
error ("%s: createFillAndTest: writing done\n",name);
|
||||||
|
for (int i=0;i<pages*PAGE_SIZE;i++)
|
||||||
|
{
|
||||||
|
if (!(i%9024) )
|
||||||
|
error ("Reading from byte %d of area %s\n",i,name);
|
||||||
|
if (i%256!=readByte(addr,i))
|
||||||
|
error ("ERROR! Byte at offset %d does not match: expected: %d, found: %d\n",i,i%256,readByte(addr,i));
|
||||||
|
}
|
||||||
|
error ("%s: createFillAndTest: reading done\n",name);
|
||||||
|
return area1;
|
||||||
|
}
|
||||||
|
catch (const char *t)
|
||||||
|
{
|
||||||
|
error ("Exception thrown! %s\n",t);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
catch (...)
|
||||||
|
{
|
||||||
|
error ("Exception thrown!\n");
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc,char **argv)
|
||||||
|
{
|
||||||
|
error ("Starting Threads!\n");
|
||||||
|
createFillAndTest(1,"myTest");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
@ -13,13 +13,18 @@ class swapFileManager {
|
|||||||
sem_id lockFreeList;
|
sem_id lockFreeList;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
// Constructors and Destructors and related
|
||||||
swapFileManager (void);
|
swapFileManager (void);
|
||||||
vnode &findNode(void); // Get an unused node
|
|
||||||
void freeVNode(vnode &); // Free a node
|
void freeVNode(vnode &); // Free a node
|
||||||
|
|
||||||
|
// Mutators
|
||||||
|
vnode &findNode(void); // Get an unused node
|
||||||
void write_block(vnode &node,void *loc,unsigned long size);
|
void write_block(vnode &node,void *loc,unsigned long size);
|
||||||
void read_block(vnode &node,void *loc,unsigned long size);
|
void read_block(vnode &node,void *loc,unsigned long size);
|
||||||
void lock() {acquire_sem(lockFreeList);}
|
void lock() {acquire_sem(lockFreeList);}
|
||||||
void unlock() {release_sem(lockFreeList);}
|
void unlock() {release_sem(lockFreeList);}
|
||||||
|
|
||||||
|
// Accessors
|
||||||
int getFD(void) {return swapFile;}
|
int getFD(void) {return swapFile;}
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
@ -39,18 +39,8 @@ int createFillAndTest(int pages,char *name)
|
|||||||
}
|
}
|
||||||
catch (const char *t)
|
catch (const char *t)
|
||||||
{
|
{
|
||||||
vm.suspendAll();
|
error ("Exception thrown! %s\n",t);
|
||||||
thread_id me=find_thread(NULL);
|
exit(1);
|
||||||
suspend_thread(clone1);
|
|
||||||
suspend_thread(mmap1);
|
|
||||||
suspend_thread(info1);
|
|
||||||
if (loop1!=me)
|
|
||||||
suspend_thread(loop1);
|
|
||||||
if (loop2!=me)
|
|
||||||
suspend_thread(loop2);
|
|
||||||
if (loop3!=me)
|
|
||||||
suspend_thread(loop3);
|
|
||||||
debugger(t);
|
|
||||||
}
|
}
|
||||||
catch (...)
|
catch (...)
|
||||||
{
|
{
|
||||||
@ -87,6 +77,11 @@ int32 loopTest(void *parameters)
|
|||||||
snooze(params->loopSnooze);
|
snooze(params->loopSnooze);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
catch (const char *t)
|
||||||
|
{
|
||||||
|
error ("Exception thrown! %s\n",t);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
catch (...)
|
catch (...)
|
||||||
{
|
{
|
||||||
error ("Exception thrown!\n");
|
error ("Exception thrown!\n");
|
||||||
@ -125,6 +120,11 @@ int32 getInfoTest(void *parameters)
|
|||||||
snooze(params->loopSnooze);
|
snooze(params->loopSnooze);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
catch (const char *t)
|
||||||
|
{
|
||||||
|
error ("Exception thrown! %s\n",t);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
catch (...)
|
catch (...)
|
||||||
{
|
{
|
||||||
error ("Exception thrown!\n");
|
error ("Exception thrown!\n");
|
||||||
@ -160,6 +160,11 @@ int32 mmapTest (void *parameters)
|
|||||||
error ("Closed file, fd = %d\n",fd);
|
error ("Closed file, fd = %d\n",fd);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
catch (const char *t)
|
||||||
|
{
|
||||||
|
error ("Exception thrown! %s\n",t);
|
||||||
|
exit(1);
|
||||||
|
}
|
||||||
catch (...)
|
catch (...)
|
||||||
{
|
{
|
||||||
error ("Exception thrown!\n");
|
error ("Exception thrown!\n");
|
||||||
@ -207,9 +212,10 @@ int main(int argc,char **argv)
|
|||||||
loopTestParameters mmap1Params={"mmap",500000,8192,400000,1000000};
|
loopTestParameters mmap1Params={"mmap",500000,8192,400000,1000000};
|
||||||
loopTestParameters clone1Params={"clone1",200000,2,300000,400000};
|
loopTestParameters clone1Params={"clone1",200000,2,300000,400000};
|
||||||
|
|
||||||
resume_thread(loop1=spawn_thread(loopTest,"area test 1",0,&area1Params));
|
error ("Starting Threads!\n");
|
||||||
resume_thread(loop2=spawn_thread(loopTest,"area test 2",0,&area2Params));
|
//resume_thread(loop1=spawn_thread(loopTest,"area test 1",0,&area1Params));
|
||||||
resume_thread(loop3=spawn_thread(loopTest,"area test 3",0,&area3Params));
|
//resume_thread(loop2=spawn_thread(loopTest,"area test 2",0,&area2Params));
|
||||||
|
//resume_thread(loop3=spawn_thread(loopTest,"area test 3",0,&area3Params));
|
||||||
resume_thread(info1=spawn_thread(getInfoTest,"info test 1",0,&info1Params));
|
resume_thread(info1=spawn_thread(getInfoTest,"info test 1",0,&info1Params));
|
||||||
//resume_thread(mmap1=spawn_thread(mmapTest,"mmap test 1",0,&mmap1Params));
|
//resume_thread(mmap1=spawn_thread(mmapTest,"mmap test 1",0,&mmap1Params));
|
||||||
//resume_thread(clone1=spawn_thread(cloneTest,"clone test 1",0,&clone1Params));
|
//resume_thread(clone1=spawn_thread(cloneTest,"clone test 1",0,&clone1Params));
|
||||||
|
@ -3,6 +3,8 @@
|
|||||||
#ifndef _VM_TYPES
|
#ifndef _VM_TYPES
|
||||||
#define _VM_TYPES
|
#define _VM_TYPES
|
||||||
const int PAGE_SIZE = 4096;
|
const int PAGE_SIZE = 4096;
|
||||||
|
const int BITS_IN_PAGE_SIZE = 12;
|
||||||
|
const int AREA_HASH_TABLE_SIZE = 40;
|
||||||
struct vnode : public node
|
struct vnode : public node
|
||||||
{
|
{
|
||||||
int fd;
|
int fd;
|
||||||
|
@ -73,7 +73,7 @@ vmInterface::vmInterface(int pages)
|
|||||||
error ("pageManager::pageManager: No memory!\n");
|
error ("pageManager::pageManager: No memory!\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
//error ("Allocated an area. Address = %x\n",vmBlock);
|
error ("Allocated an area. Address = %x\n",vmBlock);
|
||||||
// Figure out how many pages we need
|
// Figure out how many pages we need
|
||||||
int pageCount = (sizeof(poolarea)+sizeof(poolvpage)+sizeof(poolvnode)+sizeof(pageManager)+sizeof(swapFileManager)+sizeof(cacheManager)+sizeof(vmHeaderBlock)+PAGE_SIZE-1)/PAGE_SIZE;
|
int pageCount = (sizeof(poolarea)+sizeof(poolvpage)+sizeof(poolvnode)+sizeof(pageManager)+sizeof(swapFileManager)+sizeof(cacheManager)+sizeof(vmHeaderBlock)+PAGE_SIZE-1)/PAGE_SIZE;
|
||||||
if (pageCount >=pages)
|
if (pageCount >=pages)
|
||||||
@ -81,23 +81,28 @@ vmInterface::vmInterface(int pages)
|
|||||||
error ("Hey! Go buy some ram! Trying to create a VM with fewer pages than the setup will take!\n");
|
error ("Hey! Go buy some ram! Trying to create a VM with fewer pages than the setup will take!\n");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
//error ("Need %d pages, creation calls for %d\n",pageCount,pages);
|
error ("Need %d pages, creation calls for %d\n",pageCount,pages);
|
||||||
void *currentAddress = addToPointer(vmBlock,sizeof(struct vmHeaderBlock));
|
void *currentAddress = addToPointer(vmBlock,sizeof(struct vmHeaderBlock));
|
||||||
|
vmBlock->pageMan = new (currentAddress) pageManager;
|
||||||
|
currentAddress=addToPointer(currentAddress,sizeof(pageManager));
|
||||||
|
vmBlock->pageMan->setup(addToPointer(vmBlock,PAGE_SIZE*pageCount),pages-pageCount);
|
||||||
|
error ("Set up Page Man\n");
|
||||||
vmBlock->areaPool = new (currentAddress) poolarea;
|
vmBlock->areaPool = new (currentAddress) poolarea;
|
||||||
currentAddress=addToPointer(currentAddress,sizeof(poolarea));
|
currentAddress=addToPointer(currentAddress,sizeof(poolarea));
|
||||||
vmBlock->vpagePool = new (currentAddress) poolvpage;
|
vmBlock->vpagePool = new (currentAddress) poolvpage;
|
||||||
currentAddress=addToPointer(currentAddress,sizeof(poolvpage));
|
currentAddress=addToPointer(currentAddress,sizeof(poolvpage));
|
||||||
vmBlock->vnodePool = new (currentAddress) poolvnode;
|
vmBlock->vnodePool = new (currentAddress) poolvnode;
|
||||||
currentAddress=addToPointer(currentAddress,sizeof(poolvnode));
|
currentAddress=addToPointer(currentAddress,sizeof(poolvnode));
|
||||||
vmBlock->pageMan = new (currentAddress) pageManager;
|
|
||||||
currentAddress=addToPointer(currentAddress,sizeof(pageManager));
|
|
||||||
vmBlock->swapMan = new (currentAddress) swapFileManager;
|
vmBlock->swapMan = new (currentAddress) swapFileManager;
|
||||||
currentAddress=addToPointer(currentAddress,sizeof(swapFileManager));
|
currentAddress=addToPointer(currentAddress,sizeof(swapFileManager));
|
||||||
vmBlock->cacheMan = new (currentAddress) cacheManager;
|
vmBlock->cacheMan = new (currentAddress) cacheManager;
|
||||||
currentAddress=addToPointer(currentAddress,sizeof(cacheManager));
|
currentAddress=addToPointer(currentAddress,sizeof(cacheManager));
|
||||||
//error ("Need %d pages, creation calls for %d\n",pageCount,pages);
|
error ("Need %d pages, creation calls for %d\n",pageCount,pages);
|
||||||
//error ("vmBlock is at %x, end of structures is at %x, pageMan called with address %x, pages = %d\n",vmBlock,currentAddress,addToPointer(vmBlock,PAGE_SIZE*pageCount),pages-pageCount);
|
error ("vmBlock is at %x, end of structures is at %x, pageMan called with address %x, pages = %d\n",vmBlock,currentAddress,addToPointer(vmBlock,PAGE_SIZE*pageCount),pages-pageCount);
|
||||||
vmBlock->pageMan->setup(addToPointer(vmBlock,PAGE_SIZE*pageCount),pages-pageCount);
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
error ("Area found!\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
resume_thread(tid_cleaner=spawn_thread(cleanerThread,"cleanerThread",0,(vmBlock->pageMan)));
|
resume_thread(tid_cleaner=spawn_thread(cleanerThread,"cleanerThread",0,(vmBlock->pageMan)));
|
||||||
@ -214,10 +219,3 @@ status_t vmInterface::munmap(void *addr, size_t len)
|
|||||||
retVal = getAM()->munmap(addr,len);
|
retVal = getAM()->munmap(addr,len);
|
||||||
return retVal;
|
return retVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
void vmInterface::suspendAll(void)
|
|
||||||
{
|
|
||||||
suspend_thread(tid_cleaner);
|
|
||||||
suspend_thread(tid_saver);
|
|
||||||
suspend_thread(tid_pager);
|
|
||||||
}
|
|
||||||
|
@ -33,6 +33,5 @@ class vmInterface // This is the class that "owns" all of the managers.
|
|||||||
void setByte(unsigned long offset,char value) {getAM()->setByte(offset,value);} // This is for testing only
|
void setByte(unsigned long offset,char value) {getAM()->setByte(offset,value);} // This is for testing only
|
||||||
int getInt(unsigned long offset) {return getAM()->getInt(offset);} // This is for testing only
|
int getInt(unsigned long offset) {return getAM()->getInt(offset);} // This is for testing only
|
||||||
void setInt(unsigned long offset,int value) {getAM()->setByte(offset,value);} // This is for testing only
|
void setInt(unsigned long offset,int value) {getAM()->setByte(offset,value);} // This is for testing only
|
||||||
void suspendAll(void);
|
|
||||||
thread_id tid_cleaner,tid_saver,tid_pager;
|
thread_id tid_cleaner,tid_saver,tid_pager;
|
||||||
};
|
};
|
||||||
|
18
src/kernel/vm2/vnodeManager.C
Normal file
18
src/kernel/vm2/vnodeManager.C
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
#include <vnodeManager.h>
|
||||||
|
|
||||||
|
vpage *vnodeManager::findVnode(vnode &target)
|
||||||
|
{
|
||||||
|
managedVnode *found=vnodes.find(&target);
|
||||||
|
if (found==NULL)
|
||||||
|
return NULL;
|
||||||
|
else
|
||||||
|
return found->pages.peek();
|
||||||
|
}
|
||||||
|
|
||||||
|
void vnodeManager::addVnode(vnode &target,vpage &vp)
|
||||||
|
{
|
||||||
|
// Allocate space for a managed vnode
|
||||||
|
managedVnode *mv = NULL; // Fill this in later - maybe another pool?
|
||||||
|
mv->node=⌖
|
||||||
|
mv->pages.add(&vp);
|
||||||
|
}
|
33
src/kernel/vm2/vnodeManager.h
Normal file
33
src/kernel/vm2/vnodeManager.h
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#include <hashTable.h>
|
||||||
|
|
||||||
|
// vnode manager tracks which vnodes are already mapped to pages
|
||||||
|
// and facilitates sharing of memory containing the same disk space.
|
||||||
|
|
||||||
|
struct managedVnode
|
||||||
|
{
|
||||||
|
vnode *node;
|
||||||
|
list pages; // Hold the list of vpages that this vnode points to
|
||||||
|
};
|
||||||
|
|
||||||
|
ulong mnHash (node &vnodule) { return ((managedNode &)vnodule).node->fd; }
|
||||||
|
bool mnIsEqual (node &vnodule1,&vnodule2) {
|
||||||
|
managedNode &v1= ((managedNode &)vnodule1);
|
||||||
|
managedNode &v2= ((managedNode &)vnodule2);
|
||||||
|
return v1.node->fd==v2.node->fd && v1.node->offset==v2.node->offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
class vnodeManager
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
vnodeManager(void) : vnodes(20)
|
||||||
|
{
|
||||||
|
vnodes.setHash(mnHash);
|
||||||
|
vnodes.setIsEqual(mnIsEqual);
|
||||||
|
}
|
||||||
|
|
||||||
|
vpage *findVnode(vnode &target); // pass in a vnode, get back the "master" vpage
|
||||||
|
void addVnode (vnode &target, vpage &vp);
|
||||||
|
|
||||||
|
private:
|
||||||
|
hashTable vnodes;
|
||||||
|
}
|
@ -8,10 +8,8 @@
|
|||||||
|
|
||||||
extern vmHeaderBlock *vmBlock;
|
extern vmHeaderBlock *vmBlock;
|
||||||
|
|
||||||
void vpage::flush(void)
|
void vpage::flush(void) {
|
||||||
{
|
if (protection==writable && dirty) {
|
||||||
if (protection==writable && dirty)
|
|
||||||
{
|
|
||||||
//error (vpage::write_block: writing, backingNode->fd = %d, backingNode->offset = %d, address = %x\n",backingNode->fd, backingNode->offset,loc);
|
//error (vpage::write_block: writing, backingNode->fd = %d, backingNode->offset = %d, address = %x\n",backingNode->fd, backingNode->offset,loc);
|
||||||
if (-1==lseek(backingNode->fd,backingNode->offset,SEEK_SET))
|
if (-1==lseek(backingNode->fd,backingNode->offset,SEEK_SET))
|
||||||
error ("vpage::flush:seek failed, fd = %d, errno = %d, %s\n",backingNode->fd,errno,strerror(errno));
|
error ("vpage::flush:seek failed, fd = %d, errno = %d, %s\n",backingNode->fd,errno,strerror(errno));
|
||||||
@ -22,8 +20,7 @@ void vpage::flush(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void vpage::refresh(void)
|
void vpage::refresh(void) {
|
||||||
{
|
|
||||||
if (backingNode->valid==false)
|
if (backingNode->valid==false)
|
||||||
return; // Do nothing. This prevents "garbage" data on disk from being read in...
|
return; // Do nothing. This prevents "garbage" data on disk from being read in...
|
||||||
//error ("vpage::refresh: reading, backingNode->fd = %d, backingNode->offset = %d into %x\n",backingNode->fd, backingNode->offset,loc);
|
//error ("vpage::refresh: reading, backingNode->fd = %d, backingNode->offset = %d into %x\n",backingNode->fd, backingNode->offset,loc);
|
||||||
@ -33,27 +30,19 @@ void vpage::refresh(void)
|
|||||||
error ("vpage::refresh: failed, fd = %d, errno = %d, %s\n",backingNode->fd,errno,strerror(errno));
|
error ("vpage::refresh: failed, fd = %d, errno = %d, %s\n",backingNode->fd,errno,strerror(errno));
|
||||||
}
|
}
|
||||||
|
|
||||||
vpage::vpage(void)
|
vpage::vpage(void) : physPage(NULL),backingNode(NULL),protection(none),dirty(false),swappable(false),start_address(0),end_address(0)
|
||||||
{
|
{
|
||||||
physPage=NULL;
|
|
||||||
backingNode=NULL;
|
|
||||||
protection=none;
|
|
||||||
dirty=false;
|
|
||||||
swappable=false;
|
|
||||||
start_address=end_address=0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// backing and/or physMem can be NULL/0.
|
// backing and/or physMem can be NULL/0.
|
||||||
void vpage::setup(unsigned long start,vnode *backing, page *physMem,protectType prot,pageState state)
|
void vpage::setup(unsigned long start,vnode *backing, page *physMem,protectType prot,pageState state) {
|
||||||
{
|
error ("vpage::setup: start = %x, vnode.fd=%d, vnode.offset=%d, physMem = %x\n",start,((backing)?backing->fd:0),((backing)?backing->offset:0), ((physMem)?(physMem->getAddress()):0));
|
||||||
//error ("vpage::vpage: start = %x, vnode.fd=%d, vnode.offset=%d, physMem = %x\n",start,((backing)?backing->fd:0),((backing)?backing->offset:0), ((physMem)?(physMem->getAddress()):0));
|
|
||||||
start_address=start;
|
start_address=start;
|
||||||
end_address=start+PAGE_SIZE-1;
|
end_address=start+PAGE_SIZE-1;
|
||||||
protection=prot;
|
protection=prot;
|
||||||
swappable=(state==NO_LOCK);
|
swappable=(state==NO_LOCK);
|
||||||
|
|
||||||
if (backing)
|
if (backing) {
|
||||||
{
|
|
||||||
backingNode=backing;
|
backingNode=backing;
|
||||||
atomic_add(&(backing->count),1);
|
atomic_add(&(backing->count),1);
|
||||||
}
|
}
|
||||||
@ -61,47 +50,40 @@ void vpage::setup(unsigned long start,vnode *backing, page *physMem,protectType
|
|||||||
backingNode=&(vmBlock->swapMan->findNode());
|
backingNode=&(vmBlock->swapMan->findNode());
|
||||||
if (!physMem && (state!=LAZY) && (state!=NO_LOCK))
|
if (!physMem && (state!=LAZY) && (state!=NO_LOCK))
|
||||||
physPage=vmBlock->pageMan->getPage();
|
physPage=vmBlock->pageMan->getPage();
|
||||||
else
|
else {
|
||||||
{
|
|
||||||
if (physMem)
|
if (physMem)
|
||||||
atomic_add(&(physMem->count),1);
|
atomic_add(&(physMem->count),1);
|
||||||
physPage=physMem;
|
physPage=physMem;
|
||||||
}
|
}
|
||||||
dirty=false;
|
dirty=false;
|
||||||
//error ("vpage::vpage: ended : start = %x, vnode.fd=%d, vnode.offset=%d, physMem = %x\n",start,((backing)?backing->fd:0),((backing)?backing->offset:0), ((physMem)?(physMem->getAddress()):0));
|
error ("vpage::setup: ended : start = %x, vnode.fd=%d, vnode.offset=%d, physMem = %x\n",start,((backing)?backing->fd:0),((backing)?backing->offset:0), ((physMem)?(physMem->getAddress()):0));
|
||||||
}
|
}
|
||||||
|
|
||||||
void vpage::cleanup(void)
|
void vpage::cleanup(void) {
|
||||||
{
|
|
||||||
if (physPage) // Note that free means release one reference
|
if (physPage) // Note that free means release one reference
|
||||||
vmBlock->pageMan->freePage(physPage);
|
vmBlock->pageMan->freePage(physPage);
|
||||||
if (backingNode)
|
if (backingNode) {
|
||||||
{
|
|
||||||
if (backingNode->fd)
|
if (backingNode->fd)
|
||||||
if (backingNode->fd==vmBlock->swapMan->getFD())
|
if (backingNode->fd==vmBlock->swapMan->getFD())
|
||||||
vmBlock->swapMan->freeVNode(*backingNode);
|
vmBlock->swapMan->freeVNode(*backingNode);
|
||||||
else
|
else
|
||||||
{
|
|
||||||
if ( atomic_add(&(backingNode->count),-1)==1)
|
if ( atomic_add(&(backingNode->count),-1)==1)
|
||||||
vmBlock->vnodePool->put(backingNode);
|
vmBlock->vnodePool->put(backingNode);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void vpage::setProtection(protectType prot)
|
void vpage::setProtection(protectType prot) {
|
||||||
{
|
|
||||||
protection=prot;
|
protection=prot;
|
||||||
// Change the hardware
|
// Change the hardware
|
||||||
}
|
}
|
||||||
|
|
||||||
bool vpage::fault(void *fault_address, bool writeError) // true = OK, false = panic.
|
// This is dispatched by the real interrupt handler, who locates us
|
||||||
{ // This is dispatched by the real interrupt handler, who locates us
|
// true = OK, false = panic.
|
||||||
|
bool vpage::fault(void *fault_address, bool writeError) {
|
||||||
error ("vpage::fault: virtual address = %lx, write = %s\n",(unsigned long) fault_address,((writeError)?"true":"false"));
|
error ("vpage::fault: virtual address = %lx, write = %s\n",(unsigned long) fault_address,((writeError)?"true":"false"));
|
||||||
if (writeError && physPage)
|
if (writeError && physPage) {
|
||||||
{
|
|
||||||
dirty=true;
|
dirty=true;
|
||||||
if (protection==copyOnWrite) // Else, this was just a "let me know when I am dirty"...
|
if (protection==copyOnWrite) { // Else, this was just a "let me know when I am dirty"...
|
||||||
{
|
|
||||||
page *newPhysPage=vmBlock->pageMan->getPage();
|
page *newPhysPage=vmBlock->pageMan->getPage();
|
||||||
if (!newPhysPage) // No room at the inn
|
if (!newPhysPage) // No room at the inn
|
||||||
return false;
|
return false;
|
||||||
@ -128,8 +110,7 @@ bool vpage::fault(void *fault_address, bool writeError) // true = OK, false = pa
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
char vpage::getByte(unsigned long address,areaManager *manager)
|
char vpage::getByte(unsigned long address,areaManager *manager) {
|
||||||
{
|
|
||||||
// error ("vpage::getByte: address = %ld\n",address );
|
// error ("vpage::getByte: address = %ld\n",address );
|
||||||
if (!physPage)
|
if (!physPage)
|
||||||
if (!manager->fault((void *)(address),false))
|
if (!manager->fault((void *)(address),false))
|
||||||
@ -138,8 +119,7 @@ char vpage::getByte(unsigned long address,areaManager *manager)
|
|||||||
return *((char *)(address-start_address+physPage->getAddress()));
|
return *((char *)(address-start_address+physPage->getAddress()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void vpage::setByte(unsigned long address,char value,areaManager *manager)
|
void vpage::setByte(unsigned long address,char value,areaManager *manager) {
|
||||||
{
|
|
||||||
// error ("vpage::setByte: address = %d, value = %d\n",address, value);
|
// error ("vpage::setByte: address = %d, value = %d\n",address, value);
|
||||||
if (!physPage)
|
if (!physPage)
|
||||||
if (!manager->fault((void *)(address),true))
|
if (!manager->fault((void *)(address),true))
|
||||||
@ -148,8 +128,7 @@ void vpage::setByte(unsigned long address,char value,areaManager *manager)
|
|||||||
// error ("vpage::setByte: physical address = %d, value = %d\n",physPage->getAddress(), *((char *)(physPage->getAddress())));
|
// error ("vpage::setByte: physical address = %d, value = %d\n",physPage->getAddress(), *((char *)(physPage->getAddress())));
|
||||||
}
|
}
|
||||||
|
|
||||||
int vpage::getInt(unsigned long address,areaManager *manager)
|
int vpage::getInt(unsigned long address,areaManager *manager) {
|
||||||
{
|
|
||||||
error ("vpage::getInt: address = %ld\n",address );
|
error ("vpage::getInt: address = %ld\n",address );
|
||||||
if (!physPage)
|
if (!physPage)
|
||||||
if (!manager->fault((void *)(address),false))
|
if (!manager->fault((void *)(address),false))
|
||||||
@ -159,22 +138,19 @@ int vpage::getInt(unsigned long address,areaManager *manager)
|
|||||||
return *((int *)(address-start_address+physPage->getAddress()));
|
return *((int *)(address-start_address+physPage->getAddress()));
|
||||||
}
|
}
|
||||||
|
|
||||||
void vpage::setInt(unsigned long address,int value,areaManager *manager)
|
void vpage::setInt(unsigned long address,int value,areaManager *manager) {
|
||||||
{
|
|
||||||
if (!physPage)
|
if (!physPage)
|
||||||
if (!manager->fault((void *)(address),true))
|
if (!manager->fault((void *)(address),true))
|
||||||
throw ("vpage::setInt");
|
throw ("vpage::setInt");
|
||||||
*((int *)(address-start_address+physPage->getAddress()))=value;
|
*((int *)(address-start_address+physPage->getAddress()))=value;
|
||||||
}
|
}
|
||||||
|
|
||||||
void vpage::pager(int desperation)
|
void vpage::pager(int desperation) {
|
||||||
{
|
|
||||||
//error ("vpage::pager start desperation = %d\n",desperation);
|
//error ("vpage::pager start desperation = %d\n",desperation);
|
||||||
if (!swappable)
|
if (!swappable)
|
||||||
return;
|
return;
|
||||||
error ("vpage::pager swappable\n");
|
error ("vpage::pager swappable\n");
|
||||||
switch (desperation)
|
switch (desperation) {
|
||||||
{
|
|
||||||
case 1: return; break;
|
case 1: return; break;
|
||||||
case 2: if (!physPage || protection!=readable) return;break;
|
case 2: if (!physPage || protection!=readable) return;break;
|
||||||
case 3: if (!physPage || dirty) return;break;
|
case 3: if (!physPage || dirty) return;break;
|
||||||
@ -190,8 +166,7 @@ void vpage::pager(int desperation)
|
|||||||
physPage=NULL;
|
physPage=NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
void vpage::saver(void)
|
void vpage::saver(void) {
|
||||||
{
|
|
||||||
if (dirty)
|
if (dirty)
|
||||||
flush();
|
flush();
|
||||||
dirty=false;
|
dirty=false;
|
||||||
|
@ -16,27 +16,37 @@ class vpage : public node
|
|||||||
unsigned long start_address;
|
unsigned long start_address;
|
||||||
unsigned long end_address;
|
unsigned long end_address;
|
||||||
public:
|
public:
|
||||||
bool isMapped(void) {return (physPage);}
|
// Constructors and Destructors and related
|
||||||
bool contains(uint32 address) { return ((start_address<=address) && (end_address>=address)); }
|
|
||||||
void flush(void); // write page to vnode, if necessary
|
|
||||||
void refresh(void); // Read page back in from vnode
|
|
||||||
vpage(void);
|
vpage(void);
|
||||||
|
vpage(unsigned long address) {start_address=address-address%PAGE_SIZE;end_address=start_address+PAGE_SIZE-1;} // Only for lookups
|
||||||
// Setup should now only be called by the vpage manager...
|
// Setup should now only be called by the vpage manager...
|
||||||
void setup(unsigned long start,vnode *backing, page *physMem,protectType prot,pageState state); // backing and/or physMem can be NULL/0.
|
void setup(unsigned long start,vnode *backing, page *physMem,protectType prot,pageState state); // backing and/or physMem can be NULL/0.
|
||||||
void cleanup(void);
|
void cleanup(void);
|
||||||
|
|
||||||
|
// Mutators
|
||||||
void setProtection(protectType prot);
|
void setProtection(protectType prot);
|
||||||
|
void flush(void); // write page to vnode, if necessary
|
||||||
|
void refresh(void); // Read page back in from vnode
|
||||||
|
|
||||||
|
// Accessors
|
||||||
protectType getProtection(void) {return protection;}
|
protectType getProtection(void) {return protection;}
|
||||||
void *getStartAddress(void) {return (void *)start_address;}
|
void *getStartAddress(void) {return (void *)start_address;}
|
||||||
page *getPhysPage(void) {return physPage;}
|
page *getPhysPage(void) {return physPage;}
|
||||||
vnode *getBacking(void) {return backingNode;}
|
vnode *getBacking(void) {return backingNode;}
|
||||||
|
bool isMapped(void) {return (physPage);}
|
||||||
|
|
||||||
|
// Comparisson with others
|
||||||
|
ulong hash(void) {return start_address >> BITS_IN_PAGE_SIZE;}
|
||||||
|
bool operator==(vpage &rhs) {return rhs.start_address==start_address && rhs.end_address==end_address;}
|
||||||
|
bool contains(uint32 address) { return ((start_address<=address) && (end_address>=address)); }
|
||||||
|
|
||||||
|
// External methods for "server" type calls
|
||||||
bool fault(void *fault_address, bool writeError); // true = OK, false = panic.
|
bool fault(void *fault_address, bool writeError); // true = OK, false = panic.
|
||||||
|
|
||||||
void pager(int desperation);
|
void pager(int desperation);
|
||||||
void saver(void);
|
void saver(void);
|
||||||
|
|
||||||
void dump(void)
|
// Debugging
|
||||||
{
|
void dump(void) {
|
||||||
error ("Dumping vpage %p, address = %lx, physPage: \n",this,start_address);
|
error ("Dumping vpage %p, address = %lx, physPage: \n",this,start_address);
|
||||||
if (physPage)
|
if (physPage)
|
||||||
physPage->dump();
|
physPage->dump();
|
||||||
|
@ -7,13 +7,11 @@ class poolvpage
|
|||||||
list unused;
|
list unused;
|
||||||
sem_id inUse;
|
sem_id inUse;
|
||||||
public:
|
public:
|
||||||
poolvpage(void)
|
poolvpage(void) {
|
||||||
{
|
|
||||||
inUse = create_sem(1,"vpagepool");
|
inUse = create_sem(1,"vpagepool");
|
||||||
}
|
}
|
||||||
vpage *get(void);
|
vpage *get(void);
|
||||||
void put(vpage *in)
|
void put(vpage *in) {
|
||||||
{
|
|
||||||
acquire_sem(inUse);
|
acquire_sem(inUse);
|
||||||
unused.add((node *)in);
|
unused.add((node *)in);
|
||||||
release_sem(inUse);
|
release_sem(inUse);
|
||||||
|
Loading…
Reference in New Issue
Block a user