2007-07-20 14:36:31 +00:00
|
|
|
#include "sys/mman.h"
|
|
|
|
#include "sys/types.h"
|
|
|
|
#include "sys/stat.h"
|
|
|
|
#include "fcntl.h"
|
|
|
|
|
|
|
|
#include "system.h"
|
2007-08-10 23:45:47 +00:00
|
|
|
#include "finder.h"
|
2007-07-20 14:36:31 +00:00
|
|
|
|
|
|
|
using namespace vm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
const char*
|
2007-08-10 23:45:47 +00:00
|
|
|
append(System* s, const char* a, const char* b, const char* c)
|
2007-07-20 14:36:31 +00:00
|
|
|
{
|
|
|
|
unsigned al = strlen(a);
|
|
|
|
unsigned bl = strlen(b);
|
|
|
|
unsigned cl = strlen(c);
|
2007-08-10 23:45:47 +00:00
|
|
|
char* p = static_cast<char*>(s->allocate(al + bl + cl + 1));
|
2007-07-20 14:36:31 +00:00
|
|
|
memcpy(p, a, al);
|
|
|
|
memcpy(p + al, b, bl);
|
2007-08-10 23:45:47 +00:00
|
|
|
memcpy(p + al + bl, c, cl + 1);
|
2007-07-20 14:36:31 +00:00
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2007-08-27 13:46:17 +00:00
|
|
|
const char*
|
|
|
|
copy(System* s, const char* a)
|
|
|
|
{
|
|
|
|
unsigned al = strlen(a);
|
|
|
|
char* p = static_cast<char*>(s->allocate(al + 1));
|
|
|
|
memcpy(p, a, al + 1);
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2007-07-20 14:36:31 +00:00
|
|
|
const char**
|
|
|
|
parsePath(System* s, const char* path)
|
|
|
|
{
|
|
|
|
class Tokenizer {
|
|
|
|
public:
|
|
|
|
class Token {
|
|
|
|
public:
|
|
|
|
Token(const char* s, unsigned length): s(s), length(length) { }
|
|
|
|
|
|
|
|
const char* s;
|
|
|
|
unsigned length;
|
|
|
|
};
|
|
|
|
|
|
|
|
Tokenizer(const char* s, char delimiter): s(s), delimiter(delimiter) { }
|
|
|
|
|
|
|
|
bool hasMore() {
|
|
|
|
while (*s == delimiter) ++s;
|
|
|
|
return *s;
|
|
|
|
}
|
|
|
|
|
|
|
|
Token next() {
|
|
|
|
const char* p = s;
|
|
|
|
while (*s and *s != delimiter) ++s;
|
|
|
|
return Token(p, s - p);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* s;
|
|
|
|
char delimiter;
|
|
|
|
};
|
|
|
|
|
|
|
|
unsigned count = 0;
|
|
|
|
for (Tokenizer t(path, ':'); t.hasMore(); t.next()) ++ count;
|
|
|
|
|
|
|
|
const char** v = static_cast<const char**>
|
|
|
|
(s->allocate((count + 1) * sizeof(const char*)));
|
|
|
|
|
|
|
|
unsigned i = 0;
|
|
|
|
for (Tokenizer t(path, ':'); t.hasMore(); ++i) {
|
|
|
|
Tokenizer::Token token(t.next());
|
|
|
|
char* p = static_cast<char*>(s->allocate(token.length + 1));
|
|
|
|
memcpy(p, token.s, token.length);
|
|
|
|
p[token.length] = 0;
|
|
|
|
v[i] = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
v[i] = 0;
|
|
|
|
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
2007-08-10 23:45:47 +00:00
|
|
|
class MyFinder: public Finder {
|
2007-07-20 14:36:31 +00:00
|
|
|
public:
|
2007-08-10 23:45:47 +00:00
|
|
|
MyFinder(System* system, const char* path):
|
2007-07-20 14:36:31 +00:00
|
|
|
system(system),
|
2007-08-27 13:46:17 +00:00
|
|
|
path_(parsePath(system, path)),
|
|
|
|
pathString(copy(system, path))
|
2007-07-20 14:36:31 +00:00
|
|
|
{ }
|
|
|
|
|
2007-08-10 23:45:47 +00:00
|
|
|
class Data: public Finder::Data {
|
2007-07-20 14:36:31 +00:00
|
|
|
public:
|
|
|
|
Data(System* system, uint8_t* start, size_t length):
|
|
|
|
system(system),
|
|
|
|
start_(start),
|
|
|
|
length_(length)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual const uint8_t* start() {
|
|
|
|
return start_;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual size_t length() {
|
|
|
|
return length_;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void dispose() {
|
|
|
|
if (start_) {
|
|
|
|
munmap(start_, length_);
|
|
|
|
}
|
|
|
|
system->free(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
System* system;
|
|
|
|
uint8_t* start_;
|
|
|
|
size_t length_;
|
|
|
|
};
|
|
|
|
|
2007-08-10 23:45:47 +00:00
|
|
|
virtual Data* find(const char* name) {
|
2007-07-20 14:36:31 +00:00
|
|
|
Data* d = new (system->allocate(sizeof(Data))) Data(system, 0, 0);
|
|
|
|
|
2007-08-27 13:46:17 +00:00
|
|
|
for (const char** p = path_; *p; ++p) {
|
2007-08-10 23:45:47 +00:00
|
|
|
const char* file = append(system, *p, "/", name);
|
2007-07-20 14:36:31 +00:00
|
|
|
int fd = open(file, O_RDONLY);
|
|
|
|
system->free(file);
|
|
|
|
|
|
|
|
if (fd != -1) {
|
|
|
|
struct stat s;
|
|
|
|
int r = fstat(fd, &s);
|
|
|
|
if (r != -1) {
|
|
|
|
void* data = mmap(0, s.st_size, PROT_READ, MAP_PRIVATE, fd, 0);
|
|
|
|
if (data) {
|
|
|
|
d->start_ = static_cast<uint8_t*>(data);
|
|
|
|
d->length_ = s.st_size;
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
system->free(d);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-08-10 23:45:47 +00:00
|
|
|
virtual bool exists(const char* name) {
|
2007-08-27 13:46:17 +00:00
|
|
|
for (const char** p = path_; *p; ++p) {
|
2007-08-10 23:45:47 +00:00
|
|
|
const char* file = append(system, *p, "/", name);
|
|
|
|
struct stat s;
|
|
|
|
int r = stat(file, &s);
|
|
|
|
system->free(file);
|
|
|
|
if (r == 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2007-08-27 13:46:17 +00:00
|
|
|
virtual const char* path() {
|
|
|
|
return pathString;
|
|
|
|
}
|
|
|
|
|
2007-07-20 14:36:31 +00:00
|
|
|
virtual void dispose() {
|
2007-08-27 13:46:17 +00:00
|
|
|
for (const char** p = path_; *p; ++p) {
|
2007-07-20 14:36:31 +00:00
|
|
|
system->free(*p);
|
|
|
|
}
|
2007-08-27 13:46:17 +00:00
|
|
|
system->free(path_);
|
|
|
|
system->free(pathString);
|
2007-07-20 14:36:31 +00:00
|
|
|
system->free(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
System* system;
|
2007-08-27 13:46:17 +00:00
|
|
|
const char** path_;
|
|
|
|
const char* pathString;
|
2007-07-20 14:36:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
2007-08-10 23:45:47 +00:00
|
|
|
Finder*
|
|
|
|
makeFinder(System* s, const char* path)
|
2007-07-20 14:36:31 +00:00
|
|
|
{
|
2007-08-10 23:45:47 +00:00
|
|
|
return new (s->allocate(sizeof(MyFinder))) MyFinder(s, path);
|
2007-07-20 14:36:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vm
|