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:
Michael Phipps 2002-11-19 04:05:16 +00:00
parent c773cf76b7
commit bc6600175c
29 changed files with 684 additions and 456 deletions

View File

@ -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 ;

View File

@ -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;

View File

@ -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

View File

@ -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();
} }

View File

@ -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

View File

@ -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);

View File

@ -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);

View File

@ -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
}; };

View File

@ -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);

View File

@ -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

View File

@ -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;
} }

View File

@ -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

View File

@ -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;

View File

@ -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

View 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);
}

View File

@ -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;

View File

@ -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;

View File

@ -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);
} }
}; };
*/ */

View 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;
}

View File

@ -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

View File

@ -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));

View File

@ -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;

View File

@ -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);
}

View File

@ -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;
}; };

View 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=&target;
mv->pages.add(&vp);
}

View 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;
}

View File

@ -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;

View File

@ -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();

View File

@ -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);