blob: 21ab93705a5fe8fea85cfd1a9019c901d2343d42 [file] [log] [blame]
//===------- LockedMap.h - A thread-safe map implementation ---------------===//
//
// N3
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#ifndef N3_LOCKED_MAP_H
#define N3_LOCKED_MAP_H
#include <map>
#include <string.h>
#include "mvm/Object.h"
#include "mvm/PrintBuffer.h"
#include "mvm/Threads/Locks.h"
#include "types.h"
#include "VMArray.h"
namespace n3 {
class Assembly;
class CLIString;
class N3;
class VMClass;
class VMCommonClass;
class VMField;
class VMObject;
class VMMethod;
class VMField;
class UTF8;
class llvm::Function;
template<class Key, class Container, class Compare, class Upcall>
class LockedMap : public mvm::Object {
public:
typedef typename std::map<Key, Container*, Compare>::iterator iterator;
typedef Container* (*funcCreate)(Key& V, Upcall* ass);
mvm::Lock* lock;
std::map<Key, Container*, Compare> map;
inline Container* lookupOrCreate(Key& V, Upcall* ass, funcCreate func) {
lock->lock();
iterator End = map.end();
iterator I = map.find(V);
if (I == End) {
Container* res = func(V, ass);
map.insert(std::make_pair(V, res));
lock->unlock();
return res;
} else {
lock->unlock();
return ((Container*)(I->second));
}
}
inline Container* lookupOrCreate(Key&V, Container* C) {
lock->lock();
iterator End = map.end();
iterator I = map.find(V);
if (I == End) {
map.insert(std::make_pair(V, C));
lock->unlock();
return C;
} else {
lock->unlock();
return ((Container*)(I->second));
}
}
inline Container* lookup(Key& V) {
lock->lock();
iterator End = map.end();
iterator I = map.find(V);
lock->unlock();
return I != End ? ((Container*)(I->second)) : 0;
}
inline void hash(Key& k, Container* c) {
lock->lock();
map.insert(std::make_pair(k, c));
lock->unlock();
}
virtual void tracer(size_t sz) {
//lock->markAndTrace();
for (iterator i = map.begin(), e = map.end(); i!= e; ++i) {
i->second->markAndTrace();
}
}
virtual void print(mvm::PrintBuffer* buf) {
buf->write("Hashtable<>");
}
};
class ClassNameCmp {
public:
const UTF8* name;
const UTF8* nameSpace;
ClassNameCmp(const UTF8* u, const UTF8* n) : name(u), nameSpace(n) {}
inline bool operator<(const ClassNameCmp &cmp) const {
if (name < cmp.name) return true;
else if (name > cmp.name) return false;
else return nameSpace < cmp.nameSpace;
}
};
class ClassNameMap :
public LockedMap<ClassNameCmp, VMCommonClass, std::less<ClassNameCmp>, Assembly > {
public:
static VirtualTable* VT;
static ClassNameMap* allocate() {
ClassNameMap* map = gc_new(ClassNameMap)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
class ClassTokenMap :
public LockedMap<uint32, VMCommonClass, std::less<uint32>, Assembly > {
public:
static VirtualTable* VT;
static ClassTokenMap* allocate() {
ClassTokenMap* map = gc_new(ClassTokenMap)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
class FieldTokenMap :
public LockedMap<uint32, VMField, std::less<uint32>, Assembly > {
public:
static VirtualTable* VT;
static FieldTokenMap* allocate() {
FieldTokenMap* map = gc_new(FieldTokenMap)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
class MethodTokenMap :
public LockedMap<uint32, VMMethod, std::less<uint32>, Assembly > {
public:
static VirtualTable* VT;
static MethodTokenMap* allocate() {
MethodTokenMap* map = gc_new(MethodTokenMap)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
class AssemblyMap :
public LockedMap<const UTF8*, Assembly, std::less<const UTF8*>, N3 > {
public:
static VirtualTable* VT;
static AssemblyMap* allocate() {
AssemblyMap* map = gc_new(AssemblyMap)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
class StringMap :
public LockedMap<const UTF8*, CLIString, std::less<const UTF8*>, N3 > {
public:
static VirtualTable* VT;
static StringMap* allocate() {
StringMap* map = gc_new(StringMap)();
map->lock = mvm::Lock::allocRecursive();
return map;
}
};
class FunctionMap :
public LockedMap<llvm::Function*, VMMethod, std::less<llvm::Function*>, N3 > {
public:
static VirtualTable* VT;
static FunctionMap* allocate() {
FunctionMap* map = gc_new(FunctionMap)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
class UTF8Map : public mvm::Object {
public:
typedef std::multimap<uint32, const UTF8*>::iterator iterator;
mvm::Lock* lock;
std::multimap<uint32, const UTF8*> map;
static VirtualTable* VT;
const UTF8* lookupOrCreateAsciiz(const char* asciiz);
const UTF8* lookupOrCreateReader(const uint16* buf, uint32 size);
virtual void tracer(size_t sz) {
//lock->markAndTrace();
for (iterator i = map.begin(), e = map.end(); i!= e; ++i) {
i->second->markAndTrace();
}
}
virtual void print(mvm::PrintBuffer* buf) {
buf->write("UTF8 Hashtable<>");
}
static UTF8Map* allocate() {
UTF8Map* map = gc_new(UTF8Map)();
map->lock = mvm::Lock::allocNormal();
return map;
}
};
} // end namespace n3
#endif