2008-11-23 23:58:01 +00:00
|
|
|
/* Copyright (c) 2008, Avian Contributors
|
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2008-11-21 23:20:35 +00:00
|
|
|
#include "bootimage.h"
|
|
|
|
#include "heapwalk.h"
|
|
|
|
#include "common.h"
|
|
|
|
#include "machine.h"
|
|
|
|
#include "util.h"
|
2008-11-23 23:58:01 +00:00
|
|
|
#include "assembler.h"
|
|
|
|
|
|
|
|
// since we aren't linking against libstdc++, we must implement this
|
|
|
|
// ourselves:
|
|
|
|
extern "C" void __cxa_pure_virtual(void) { abort(); }
|
2008-11-21 23:20:35 +00:00
|
|
|
|
|
|
|
using namespace vm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
bool
|
|
|
|
endsWith(const char* suffix, const char* s, unsigned length)
|
|
|
|
{
|
|
|
|
unsigned suffixLength = strlen(suffix);
|
|
|
|
return length >= suffixLength
|
|
|
|
and memcmp(suffix, s + (length - suffixLength), suffixLength) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
2008-11-28 04:58:04 +00:00
|
|
|
makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
|
|
|
|
unsigned capacity)
|
2008-11-21 23:20:35 +00:00
|
|
|
{
|
2008-11-28 04:44:04 +00:00
|
|
|
unsigned size = 0;
|
2008-11-23 23:58:01 +00:00
|
|
|
t->m->processor->compileThunks(t, image, code, &size, capacity);
|
2008-11-27 20:59:40 +00:00
|
|
|
|
|
|
|
object constants = 0;
|
|
|
|
PROTECT(t, constants);
|
|
|
|
|
|
|
|
object calls = 0;
|
|
|
|
PROTECT(t, calls);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
|
|
|
for (Finder::Iterator it(t->m->finder); it.hasMore();) {
|
|
|
|
unsigned nameSize;
|
|
|
|
const char* name = it.next(&nameSize);
|
|
|
|
|
|
|
|
if (endsWith(".class", name, nameSize)) {
|
2008-11-28 18:33:11 +00:00
|
|
|
//fprintf(stderr, "%.*s\n", nameSize - 6, name);
|
2008-11-21 23:20:35 +00:00
|
|
|
object c = resolveClass
|
2008-11-28 04:44:04 +00:00
|
|
|
(t, makeByteArray(t, "%.*s", nameSize - 6, name));
|
2008-11-21 23:20:35 +00:00
|
|
|
PROTECT(t, c);
|
2008-11-28 04:44:04 +00:00
|
|
|
|
|
|
|
if (classMethodTable(t, c)) {
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, classMethodTable(t, c)); ++i) {
|
|
|
|
object method = arrayBody(t, classMethodTable(t, c), i);
|
2008-11-30 04:58:09 +00:00
|
|
|
if (methodCode(t, method) or (methodFlags(t, method) & ACC_NATIVE)) {
|
2008-11-28 04:44:04 +00:00
|
|
|
t->m->processor->compileMethod
|
2008-11-28 04:58:04 +00:00
|
|
|
(t, zone, code, &size, capacity, &constants, &calls, method);
|
2008-11-28 04:44:04 +00:00
|
|
|
}
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-27 20:59:40 +00:00
|
|
|
for (; calls; calls = tripleThird(t, calls)) {
|
|
|
|
static_cast<ListenPromise*>(pointerValue(t, tripleSecond(t, calls)))
|
|
|
|
->listener->resolve(methodCompiled(t, tripleFirst(t, calls)));
|
|
|
|
}
|
|
|
|
|
2008-11-21 23:20:35 +00:00
|
|
|
image->codeSize = size;
|
|
|
|
|
2008-11-27 20:59:40 +00:00
|
|
|
return constants;
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
objectSize(Thread* t, object o)
|
|
|
|
{
|
|
|
|
assert(t, not objectExtended(t, o));
|
|
|
|
return baseSize(t, o, objectClass(t, o));
|
|
|
|
}
|
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
void
|
2008-11-28 18:33:11 +00:00
|
|
|
visitRoots(Thread* t, BootImage* image, HeapWalker* w, object constants)
|
2008-11-23 23:58:01 +00:00
|
|
|
{
|
2008-11-28 18:33:11 +00:00
|
|
|
Machine* m = t->m;
|
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
image->loader = w->visitRoot(m->loader);
|
|
|
|
image->stringMap = w->visitRoot(m->stringMap);
|
|
|
|
image->types = w->visitRoot(m->types);
|
|
|
|
|
|
|
|
m->processor->visitRoots(image, w);
|
2008-11-28 18:33:11 +00:00
|
|
|
|
|
|
|
for (; constants; constants = tripleThird(t, constants)) {
|
|
|
|
w->visitRoot(tripleFirst(t, constants));
|
|
|
|
}
|
2008-11-23 23:58:01 +00:00
|
|
|
}
|
|
|
|
|
2008-11-30 01:39:42 +00:00
|
|
|
void
|
|
|
|
visitReference(Thread* t, HeapWalker* w, uintptr_t* heap, uintptr_t* map,
|
|
|
|
object r)
|
|
|
|
{
|
|
|
|
int target = w->map()->find(jreferenceTarget(t, r));
|
|
|
|
assert(t, target > 0);
|
|
|
|
|
|
|
|
int reference = w->map()->find(r);
|
|
|
|
assert(t, reference > 0);
|
|
|
|
|
|
|
|
unsigned index = reference - 1 + (JreferenceTarget / BytesPerWord);
|
|
|
|
markBit(map, index);
|
|
|
|
heap[index] = target;
|
|
|
|
}
|
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
HeapWalker*
|
2008-11-21 23:20:35 +00:00
|
|
|
makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map,
|
2008-11-28 18:33:11 +00:00
|
|
|
unsigned capacity, object constants)
|
2008-11-21 23:20:35 +00:00
|
|
|
{
|
2008-11-23 23:58:01 +00:00
|
|
|
class Visitor: public HeapVisitor {
|
2008-11-21 23:20:35 +00:00
|
|
|
public:
|
2008-11-23 23:58:01 +00:00
|
|
|
Visitor(Thread* t, uintptr_t* heap, uintptr_t* map, unsigned capacity):
|
2008-11-29 23:08:14 +00:00
|
|
|
t(t), current(0), heap(heap), map(map), position(0), capacity(capacity)
|
2008-11-21 23:20:35 +00:00
|
|
|
{ }
|
|
|
|
|
2008-11-28 04:44:04 +00:00
|
|
|
void visit(unsigned number) {
|
2008-11-29 23:08:14 +00:00
|
|
|
if (current) {
|
|
|
|
if (number) markBit(map, current - 1);
|
|
|
|
heap[current - 1] = number;
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void root() {
|
2008-11-29 23:08:14 +00:00
|
|
|
current = 0;
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual unsigned visitNew(object p) {
|
|
|
|
if (p) {
|
|
|
|
unsigned size = objectSize(t, p);
|
|
|
|
assert(t, position + size < capacity);
|
|
|
|
|
|
|
|
memcpy(heap + position, p, size * BytesPerWord);
|
|
|
|
|
|
|
|
unsigned number = position + 1;
|
|
|
|
position += size;
|
|
|
|
|
2008-11-28 04:44:04 +00:00
|
|
|
visit(number);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
|
|
|
return number;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void visitOld(object, unsigned number) {
|
2008-11-28 04:44:04 +00:00
|
|
|
visit(number);
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
2008-11-29 23:08:14 +00:00
|
|
|
virtual void push(object, unsigned number, unsigned offset) {
|
|
|
|
current = number + offset;
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void pop() {
|
2008-11-29 23:08:14 +00:00
|
|
|
current = 0;
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Thread* t;
|
2008-11-29 23:08:14 +00:00
|
|
|
unsigned current;
|
2008-11-21 23:20:35 +00:00
|
|
|
uintptr_t* heap;
|
|
|
|
uintptr_t* map;
|
|
|
|
unsigned position;
|
|
|
|
unsigned capacity;
|
2008-11-23 23:58:01 +00:00
|
|
|
} visitor(t, heap, map, capacity / BytesPerWord);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
HeapWalker* w = makeHeapWalker(t, &visitor);
|
2008-11-28 18:33:11 +00:00
|
|
|
visitRoots(t, image, w, constants);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-30 01:39:42 +00:00
|
|
|
for (object r = t->m->weakReferences; r; r = jreferenceVmNext(t, r)) {
|
|
|
|
visitReference(t, w, heap, map, r);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (object r = t->m->tenuredWeakReferences; r; r = jreferenceVmNext(t, r)) {
|
|
|
|
visitReference(t, w, heap, map, r);
|
|
|
|
}
|
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
image->heapSize = visitor.position * BytesPerWord;
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
return w;
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2008-11-27 20:59:40 +00:00
|
|
|
updateConstants(Thread* t, object constants, uint8_t* code, uintptr_t* codeMap,
|
2008-11-21 23:20:35 +00:00
|
|
|
HeapMap* heapTable)
|
|
|
|
{
|
2008-11-27 20:59:40 +00:00
|
|
|
for (; constants; constants = tripleThird(t, constants)) {
|
2008-11-29 23:08:14 +00:00
|
|
|
unsigned target = heapTable->find(tripleFirst(t, constants));
|
|
|
|
assert(t, target > 0);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-27 20:59:40 +00:00
|
|
|
void* dst = static_cast<ListenPromise*>
|
|
|
|
(pointerValue(t, tripleSecond(t, constants)))->listener->resolve(target);
|
|
|
|
|
|
|
|
assert(t, reinterpret_cast<intptr_t>(dst)
|
|
|
|
>= reinterpret_cast<intptr_t>(code));
|
|
|
|
|
|
|
|
markBit(codeMap, reinterpret_cast<intptr_t>(dst)
|
|
|
|
- reinterpret_cast<intptr_t>(code));
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
offset(object a, uintptr_t* b)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<uintptr_t>(b) - reinterpret_cast<uintptr_t>(a);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2008-11-28 18:33:11 +00:00
|
|
|
writeBootImage(Thread* t, FILE* out)
|
2008-11-21 23:20:35 +00:00
|
|
|
{
|
2008-11-28 04:58:04 +00:00
|
|
|
Zone zone(t->m->system, t->m->heap, 64 * 1024);
|
2008-11-21 23:20:35 +00:00
|
|
|
BootImage image;
|
|
|
|
|
|
|
|
const unsigned CodeCapacity = 32 * 1024 * 1024;
|
|
|
|
uint8_t* code = static_cast<uint8_t*>(t->m->heap->allocate(CodeCapacity));
|
|
|
|
uintptr_t* codeMap = static_cast<uintptr_t*>
|
|
|
|
(t->m->heap->allocate(codeMapSize(CodeCapacity)));
|
|
|
|
memset(codeMap, 0, codeMapSize(CodeCapacity));
|
|
|
|
|
2008-11-28 04:58:04 +00:00
|
|
|
object constants = makeCodeImage(t, &zone, &image, code, CodeCapacity);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
|
|
|
const unsigned HeapCapacity = 32 * 1024 * 1024;
|
|
|
|
uintptr_t* heap = static_cast<uintptr_t*>
|
|
|
|
(t->m->heap->allocate(HeapCapacity));
|
|
|
|
uintptr_t* heapMap = static_cast<uintptr_t*>
|
|
|
|
(t->m->heap->allocate(heapMapSize(HeapCapacity)));
|
|
|
|
memset(heapMap, 0, heapMapSize(HeapCapacity));
|
|
|
|
|
2008-11-30 01:39:42 +00:00
|
|
|
PROTECT(t, constants);
|
|
|
|
collect(t, Heap::MajorCollection);
|
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
HeapWalker* heapWalker = makeHeapImage
|
2008-11-28 18:33:11 +00:00
|
|
|
(t, &image, heap, heapMap, HeapCapacity, constants);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-27 20:59:40 +00:00
|
|
|
updateConstants(t, constants, code, codeMap, heapWalker->map());
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
heapWalker->dispose();
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-23 23:58:01 +00:00
|
|
|
image.magic = BootImage::Magic;
|
2008-11-30 01:39:42 +00:00
|
|
|
image.codeBase = reinterpret_cast<uintptr_t>(code);
|
|
|
|
|
|
|
|
fprintf(stderr, "heap size %d code size %d\n",
|
|
|
|
image.heapSize, image.codeSize);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-30 01:39:42 +00:00
|
|
|
if (true) {
|
2008-11-28 22:02:45 +00:00
|
|
|
fwrite(&image, sizeof(BootImage), 1, out);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-28 22:02:45 +00:00
|
|
|
fwrite(heapMap, pad(heapMapSize(image.heapSize)), 1, out);
|
|
|
|
fwrite(heap, pad(image.heapSize), 1, out);
|
2008-11-21 23:20:35 +00:00
|
|
|
|
2008-11-28 22:02:45 +00:00
|
|
|
fwrite(codeMap, pad(codeMapSize(image.codeSize)), 1, out);
|
|
|
|
fwrite(code, pad(image.codeSize), 1, out);
|
|
|
|
}
|
2008-11-21 23:20:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int ac, const char** av)
|
|
|
|
{
|
|
|
|
if (ac != 2) {
|
|
|
|
fprintf(stderr, "usage: %s <classpath>\n", av[0]);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
System* s = makeSystem(0);
|
|
|
|
Heap* h = makeHeap(s, 128 * 1024 * 1024);
|
2008-11-28 04:44:04 +00:00
|
|
|
Finder* f = makeFinder(s, av[1], 0);
|
2008-11-21 23:20:35 +00:00
|
|
|
Processor* p = makeProcessor(s, h);
|
|
|
|
Machine* m = new (h->allocate(sizeof(Machine))) Machine(s, h, f, p, 0, 0);
|
|
|
|
Thread* t = p->makeThread(m, 0, 0);
|
|
|
|
|
|
|
|
enter(t, Thread::ActiveState);
|
|
|
|
enter(t, Thread::IdleState);
|
|
|
|
|
|
|
|
writeBootImage(t, stdout);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|