From 5b4f17997f05321ddcb559baa45da18185de6bf5 Mon Sep 17 00:00:00 2001 From: Joel Dice Date: Mon, 29 Aug 2011 19:00:17 -0600 Subject: [PATCH 1/5] progress towards cross-architecture bootimage builds This monster commit is the first step towards supporting cross-architecture bootimage builds. The challenge is to build a heap and code image for the target platform where the word size and endianess may differ from those of the build architecture. That means the memory layout of objects may differ due to alignment and size differences, so we can't just copy objects into the heap image unchanged; we must copy field by field, resizing values, reversing endianess and shifting offsets as necessary. This commit also removes POD (plain old data) type support from the type generator because it added a lot of complication and little value. --- makefile | 34 +- src/assembler.h | 2 +- src/bootimage.cpp | 971 +++++++++++++++++++++++++++++++++++++---- src/compile.cpp | 216 ++++----- src/compiler.cpp | 206 ++++----- src/heap.cpp | 11 +- src/heap.h | 6 + src/interpret.cpp | 8 +- src/machine.cpp | 45 +- src/machine.h | 77 +++- src/process.cpp | 2 +- src/target.h | 66 +++ src/type-generator.cpp | 267 +++++------ src/types.def | 20 +- src/x86.cpp | 289 ++++++------ 15 files changed, 1567 insertions(+), 653 deletions(-) create mode 100644 src/target.h diff --git a/makefile b/makefile index 3474de80f0..852cb6a816 100755 --- a/makefile +++ b/makefile @@ -169,7 +169,8 @@ common-cflags = $(warnings) -fno-rtti -fno-exceptions \ "-I$(JAVA_HOME)/include" -idirafter $(src) -I$(build) $(classpath-cflags) \ -D__STDC_LIMIT_MACROS -D_JNI_IMPLEMENTATION_ -DAVIAN_VERSION=\"$(version)\" \ -DUSE_ATOMIC_OPERATIONS -DAVIAN_JAVA_HOME=\"$(javahome)\" \ - -DAVIAN_EMBED_PREFIX=\"$(embed-prefix)\" + -DAVIAN_EMBED_PREFIX=\"$(embed-prefix)\" \ + -DTARGET_BYTES_PER_WORD=$(pointer-size) ifneq (,$(filter i386 x86_64,$(arch))) ifeq ($(use-frame-pointer),true) @@ -208,6 +209,12 @@ shared = -shared openjdk-extra-cflags = -fvisibility=hidden +ifeq ($(build-arch),powerpc) + ifneq ($(arch),$(build-arch)) + cflags += -DTARGET_OPPOSITE_ENDIAN + endif +endif + ifeq ($(arch),i386) pointer-size = 4 endif @@ -215,6 +222,10 @@ ifeq ($(arch),powerpc) asm = powerpc pointer-size = 4 + ifneq ($(arch),$(build-arch)) + cflags += -DTARGET_OPPOSITE_ENDIAN + endif + ifneq ($(platform),darwin) ifneq ($(arch),$(build-arch)) converter-cflags += -DOPPOSITE_ENDIAN @@ -251,7 +262,13 @@ ifeq ($(arch),arm) endif endif +ifeq ($(platform),linux) + cflags += -DTARGET_PLATFORM_LINUX +endif + ifeq ($(platform),darwin) + cflags += -DTARGET_PLATFORM_DARWIN + ifeq (${OSX_SDK_SYSROOT},) OSX_SDK_SYSROOT = 10.4u endif @@ -330,6 +347,8 @@ ifeq ($(platform),darwin) endif ifeq ($(platform),windows) + cflags += -DTARGET_PLATFORM_WINDOWS + inc = "$(root)/win32/include" lib = "$(root)/win32/lib" @@ -479,7 +498,8 @@ generated-code = \ $(build)/type-constructors.cpp \ $(build)/type-initializations.cpp \ $(build)/type-java-initializations.cpp \ - $(build)/type-name-initializations.cpp + $(build)/type-name-initializations.cpp \ + $(build)/type-maps.cpp vm-depends := $(generated-code) $(wildcard $(src)/*.h) @@ -537,16 +557,6 @@ bootimage-bin = $(build)/bootimage.bin bootimage-object = $(build)/bootimage-bin.o ifeq ($(bootimage),true) - ifneq ($(build-arch),$(arch)) -$(error "bootimage cross-builds not yet supported") - endif - - ifeq ($(arch),x86_64) - ifneq ($(build-platform),$(platform)) -$(error "bootimage cross-builds not yet supported") - endif - endif - vm-classpath-object = $(bootimage-object) cflags += -DBOOT_IMAGE=\"bootimageBin\" -DAVIAN_CLASSPATH=\"\" else diff --git a/src/assembler.h b/src/assembler.h index 4805f54e80..39d0987ce0 100644 --- a/src/assembler.h +++ b/src/assembler.h @@ -343,7 +343,7 @@ class Assembler { virtual void updateCall(UnaryOperation op, void* returnAddress, void* newTarget) = 0; - virtual void setConstant(void* dst, uintptr_t constant) = 0; + virtual void setConstant(void* dst, uint64_t constant) = 0; virtual unsigned alignFrameSize(unsigned sizeInWords) = 0; diff --git a/src/bootimage.cpp b/src/bootimage.cpp index b58439e3a7..0626eb5c77 100644 --- a/src/bootimage.cpp +++ b/src/bootimage.cpp @@ -9,11 +9,14 @@ details. */ #include "bootimage.h" +#include "heap.h" #include "heapwalk.h" #include "common.h" #include "machine.h" #include "util.h" +#include "stream.h" #include "assembler.h" +#include "target.h" // since we aren't linking against libstdc++, we must implement this // ourselves: @@ -25,6 +28,95 @@ namespace { const unsigned HeapCapacity = 256 * 1024 * 1024; +const unsigned TargetFixieSizeInBytes = 8 + (TargetBytesPerWord * 2); +const unsigned TargetFixieSizeInWords = ceiling + (TargetFixieSizeInBytes, TargetBytesPerWord); +const unsigned TargetFixieAge = 0; +const unsigned TargetFixieHasMask = 1; +const unsigned TargetFixieSize = 4; + +const bool DebugNativeTarget = false; + +enum Type { + Type_none, + Type_object, + Type_int8_t, + Type_uint8_t, + Type_int16_t, + Type_uint16_t, + Type_int32_t, + Type_uint32_t, + Type_intptr_t, + Type_uintptr_t, + Type_int64_t, + Type_int64_t_pad, + Type_uint64_t, + Type_float, + Type_double, + Type_double_pad, + Type_word, + Type_array +}; + +class Field { + public: + Field(Type type, unsigned offset, unsigned targetOffset): + type(type), offset(offset), targetOffset(targetOffset) + { } + + Type type; + unsigned offset; + unsigned targetOffset; +}; + +class TypeMap { + public: + enum Kind { + NormalKind, + SingletonKind, + PoolKind + }; + + TypeMap(unsigned buildFixedSizeInWords, unsigned targetFixedSizeInWords, + unsigned fixedFieldCount, Kind kind = NormalKind, + unsigned buildArrayElementSizeInBytes = 0, + unsigned targetArrayElementSizeInBytes = 0, + Type arrayElementType = Type_none): + buildFixedSizeInWords(buildFixedSizeInWords), + targetFixedSizeInWords(targetFixedSizeInWords), + fixedFieldCount(fixedFieldCount), + buildArrayElementSizeInBytes(buildArrayElementSizeInBytes), + targetArrayElementSizeInBytes(targetArrayElementSizeInBytes), + arrayElementType(arrayElementType), + kind(kind) + { } + + uintptr_t* targetFixedOffsets() { + return reinterpret_cast(this + 1); + } + + Field* fixedFields() { + return reinterpret_cast + (targetFixedOffsets() + (buildFixedSizeInWords * BytesPerWord)); + } + + static unsigned sizeInBytes(unsigned buildFixedSizeInWords, + unsigned fixedFieldCount) + { + return sizeof(TypeMap) + + (buildFixedSizeInWords * BytesPerWord * BytesPerWord) + + (sizeof(Field) * fixedFieldCount); + } + + unsigned buildFixedSizeInWords; + unsigned targetFixedSizeInWords; + unsigned fixedFieldCount; + unsigned buildArrayElementSizeInBytes; + unsigned targetArrayElementSizeInBytes; + Type arrayElementType; + Kind kind; +}; + // Notes on immutable references in the heap image: // // One of the advantages of a bootimage-based build is that reduces @@ -66,8 +158,10 @@ endsWith(const char* suffix, const char* s, unsigned length) object makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, uintptr_t* codeMap, const char* className, - const char* methodName, const char* methodSpec) + const char* methodName, const char* methodSpec, object typeMaps) { + PROTECT(t, typeMaps); + object constants = 0; PROTECT(t, constants); @@ -76,24 +170,259 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, DelayedPromise* addresses = 0; - for (Finder::Iterator it - (static_cast - (systemClassLoaderFinder(t, root(t, Machine::BootLoader)))); - it.hasMore();) - { + Finder* finder = static_cast + (systemClassLoaderFinder(t, root(t, Machine::BootLoader))); + + for (Finder::Iterator it(finder); it.hasMore();) { unsigned nameSize = 0; const char* name = it.next(&nameSize); if (endsWith(".class", name, nameSize) and (className == 0 or strncmp(name, className, nameSize - 6) == 0)) { -// fprintf(stderr, "%.*s\n", nameSize - 6, name); + // fprintf(stderr, "%.*s\n", nameSize - 6, name); object c = resolveSystemClass (t, root(t, Machine::BootLoader), makeByteArray(t, "%.*s", nameSize - 6, name), true); PROTECT(t, c); + System::Region* region = finder->find(name); + + { THREAD_RESOURCE(t, System::Region*, region, region->dispose()); + + class Client: public Stream::Client { + public: + Client(Thread* t): t(t) { } + + virtual void NO_RETURN handleError() { + vm::abort(t); + } + + private: + Thread* t; + } client(t); + + Stream s(&client, region->start(), region->length()); + + uint32_t magic = s.read4(); + expect(t, magic == 0xCAFEBABE); + s.read2(); // minor version + s.read2(); // major version + + unsigned count = s.read2() - 1; + if (count) { + Type types[count + 2]; + types[0] = Type_object; + types[1] = Type_intptr_t; + + for (unsigned i = 2; i < count + 2; ++i) { + switch (s.read1()) { + case CONSTANT_Class: + case CONSTANT_String: + types[i] = Type_object; + s.skip(2); + break; + + case CONSTANT_Integer: + case CONSTANT_Float: + types[i] = Type_int32_t; + s.skip(4); + break; + + case CONSTANT_NameAndType: + case CONSTANT_Fieldref: + case CONSTANT_Methodref: + case CONSTANT_InterfaceMethodref: + types[i] = Type_object; + s.skip(4); + break; + + case CONSTANT_Long: + types[i++] = Type_int64_t; + types[i] = Type_int64_t_pad; + s.skip(8); + break; + + case CONSTANT_Double: + types[i++] = Type_double; + types[i] = Type_double_pad; + s.skip(8); + break; + + case CONSTANT_Utf8: + types[i] = Type_object; + s.skip(s.read2()); + break; + + default: abort(t); + } + } + + object array = makeByteArray + (t, TypeMap::sizeInBytes(count + 2, count + 2)); + + TypeMap* map = new (&byteArrayBody(t, array, 0)) TypeMap + (count + 2, count + 2, count + 2, TypeMap::PoolKind); + + for (unsigned i = 0; i < count + 2; ++i) { + expect(t, i < map->buildFixedSizeInWords); + + map->targetFixedOffsets()[i * BytesPerWord] + = i * TargetBytesPerWord; + + new (map->fixedFields() + i) Field + (types[i], i * BytesPerWord, i * TargetBytesPerWord); + } + + hashMapInsert + (t, typeMaps, hashMapFind + (t, root(t, Machine::PoolMap), c, objectHash, objectEqual), array, + objectHash); + } + } + + if (classFieldTable(t, c)) { + unsigned count = arrayLength(t, classFieldTable(t, c)); + + Type memberTypes[count + 1]; + memberTypes[0] = Type_object; + unsigned buildMemberOffsets[count + 1]; + buildMemberOffsets[0] = 0; + unsigned targetMemberOffsets[count + 1]; + targetMemberOffsets[0] = 0; + unsigned memberIndex = 1; + unsigned buildMemberOffset = BytesPerWord; + unsigned targetMemberOffset = TargetBytesPerWord; + + Type staticTypes[count + 2]; + staticTypes[0] = Type_object; + staticTypes[1] = Type_intptr_t; + unsigned buildStaticOffsets[count + 2]; + buildStaticOffsets[0] = 0; + buildStaticOffsets[1] = BytesPerWord; + unsigned targetStaticOffsets[count + 2]; + targetStaticOffsets[0] = 0; + targetStaticOffsets[1] = TargetBytesPerWord; + unsigned staticIndex = 2; + unsigned buildStaticOffset = BytesPerWord * 2; + unsigned targetStaticOffset = TargetBytesPerWord * 2; + + for (unsigned i = 0; i < count; ++i) { + object field = arrayBody(t, classFieldTable(t, c), i); + unsigned size = fieldSize(t, fieldCode(t, field)); + + Type type; + switch (fieldCode(t, field)) { + case ObjectField: + type = Type_object; + size = TargetBytesPerWord; + break; + + case ByteField: + case BooleanField: + type = Type_int8_t; + break; + + case CharField: + case ShortField: + type = Type_int8_t; + break; + + case FloatField: + case IntField: + type = Type_int32_t; + break; + + case LongField: + case DoubleField: + type = Type_int64_t; + break; + + default: abort(t); + } + + if (fieldFlags(t, field) & ACC_STATIC) { + staticTypes[staticIndex] = type; + + while (targetStaticOffset % size) { + ++ targetStaticOffset; + } + + targetStaticOffsets[staticIndex] = targetStaticOffset; + + targetStaticOffset += size; + + buildStaticOffset = fieldOffset(t, field); + buildStaticOffsets[staticIndex] = buildStaticOffset; + + ++ staticIndex; + } else { + memberTypes[memberIndex] = type; + + while (targetMemberOffset % size) { + ++ targetMemberOffset; + } + + targetMemberOffsets[memberIndex] = targetMemberOffset; + + targetMemberOffset += size; + + buildMemberOffset = fieldOffset(t, field); + buildMemberOffsets[memberIndex] = buildMemberOffset; + + ++ memberIndex; + } + } + + { object array = makeByteArray + (t, TypeMap::sizeInBytes + (ceiling(classFixedSize(t, c), BytesPerWord), memberIndex)); + + TypeMap* map = new (&byteArrayBody(t, array, 0)) TypeMap + (ceiling(classFixedSize(t, c), BytesPerWord), + ceiling(targetMemberOffset, TargetBytesPerWord), memberIndex); + + for (unsigned i = 0; i < memberIndex; ++i) { + expect(t, buildMemberOffsets[i] + < map->buildFixedSizeInWords * BytesPerWord); + + map->targetFixedOffsets()[buildMemberOffsets[i]] + = targetMemberOffsets[i]; + + new (map->fixedFields() + i) Field + (memberTypes[i], buildMemberOffsets[i], targetMemberOffsets[i]); + } + + hashMapInsert(t, typeMaps, c, array, objectHash); + } + + if (classStaticTable(t, c)) { + object array = makeByteArray + (t, TypeMap::sizeInBytes + (singletonCount(t, classStaticTable(t, c)) + 2, staticIndex)); + + TypeMap* map = new (&byteArrayBody(t, array, 0)) TypeMap + (singletonCount(t, classStaticTable(t, c)) + 2, + ceiling(targetStaticOffset, TargetBytesPerWord), staticIndex, + TypeMap::SingletonKind); + + for (unsigned i = 0; i < staticIndex; ++i) { + expect(t, buildStaticOffsets[i] + < map->buildFixedSizeInWords * BytesPerWord); + + map->targetFixedOffsets()[buildStaticOffsets[i]] + = targetStaticOffsets[i]; + + new (map->fixedFields() + i) Field + (staticTypes[i], buildStaticOffsets[i], targetStaticOffsets[i]); + } + + hashMapInsert + (t, typeMaps, classStaticTable(t, c), array, objectHash); + } + } + if (classMethodTable(t, c)) { for (unsigned i = 0; i < arrayLength(t, classMethodTable(t, c)); ++i) { object method = arrayBody(t, classMethodTable(t, c), i); @@ -163,18 +492,18 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, for (; addresses; addresses = addresses->next) { uint8_t* value = reinterpret_cast(addresses->basis->value()); - assert(t, value >= code); + expect(t, value >= code); void* location; bool flat = addresses->listener->resolve (reinterpret_cast(code), &location); - uintptr_t offset = value - code; + target_uintptr_t offset = value - code; if (flat) { offset |= BootFlatConstant; } - memcpy(location, &offset, BytesPerWord); + memcpy(location, &offset, TargetBytesPerWord); - assert(t, reinterpret_cast(location) + expect(t, reinterpret_cast(location) >= reinterpret_cast(code)); markBit(codeMap, reinterpret_cast(location) @@ -187,8 +516,6 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, unsigned objectSize(Thread* t, object o) { - assert(t, not objectExtended(t, o)); - return baseSize(t, o, objectClass(t, o)); } @@ -214,20 +541,310 @@ visitRoots(Thread* t, BootImage* image, HeapWalker* w, object constants) } } +TypeMap* +typeMap(Thread* t, object typeMaps, object p) +{ + return reinterpret_cast + (&byteArrayBody + (t, objectClass(t, p) == type(t, Machine::SingletonType) + ? hashMapFind(t, typeMaps, p, objectHash, objectEqual) + : hashMapFind(t, typeMaps, objectClass(t, p), objectHash, objectEqual), + 0)); +} + +unsigned +targetOffset(Thread* t, object typeMaps, object p, unsigned offset) +{ + TypeMap* map = typeMap(t, typeMaps, p); + + if (map->targetArrayElementSizeInBytes + and offset >= map->buildFixedSizeInWords * BytesPerWord) + { + return (map->targetFixedSizeInWords * TargetBytesPerWord) + + (((offset - (map->buildFixedSizeInWords * BytesPerWord)) + / map->buildArrayElementSizeInBytes) + * map->targetArrayElementSizeInBytes); + } else { + return map->targetFixedOffsets()[offset]; + } +} + +unsigned +targetSize(Thread* t, object typeMaps, object p) +{ + TypeMap* map = typeMap(t, typeMaps, p); + + if (map->targetArrayElementSizeInBytes) { + return map->targetFixedSizeInWords + + ceiling(map->targetArrayElementSizeInBytes + * cast + (p, (map->buildFixedSizeInWords - 1) * BytesPerWord), + TargetBytesPerWord); + } else { + switch (map->kind) { + case TypeMap::NormalKind: + return map->targetFixedSizeInWords; + + case TypeMap::SingletonKind: + return map->targetFixedSizeInWords + singletonMaskSize + (map->targetFixedSizeInWords - 2, TargetBitsPerWord); + + case TypeMap::PoolKind: { + unsigned maskSize = poolMaskSize + (map->targetFixedSizeInWords - 2, TargetBitsPerWord); + + return map->targetFixedSizeInWords + maskSize + singletonMaskSize + (map->targetFixedSizeInWords - 2 + maskSize, TargetBitsPerWord); + } + + default: abort(t); + } + } +} + +void +copy(Thread* t, uint8_t* src, uint8_t* dst, Type type) +{ + switch (type) { + case Type_int8_t: + memcpy(dst, src, 1); + break; + + case Type_int16_t: { + int16_t s; memcpy(&s, src, 2); + int16_t d = TARGET_V2(s); + memcpy(dst, &d, 2); + } break; + + case Type_int32_t: + case Type_float: { + int32_t s; memcpy(&s, src, 4); + int32_t d = TARGET_V4(s); + memcpy(dst, &d, 4); + } break; + + case Type_int64_t: + case Type_double: { + int64_t s; memcpy(&s, src, 8); + int64_t d = TARGET_V8(s); + memcpy(dst, &d, 8); + } break; + + case Type_int64_t_pad: + case Type_double_pad: + break; + + case Type_intptr_t: { + intptr_t s; memcpy(&s, src, BytesPerWord); + target_intptr_t d = TARGET_VW(s); + memcpy(dst, &d, TargetBytesPerWord); + } break; + + case Type_object: { + memset(dst, 0, TargetBytesPerWord); + } break; + + default: abort(t); + } +} + +bool +nonObjectsEqual(uint8_t* src, uint8_t* dst, Type type) +{ + switch (type) { + case Type_int8_t: + return memcmp(dst, src, 1) == 0; + + case Type_int16_t: + return memcmp(dst, src, 2) == 0; + + case Type_int32_t: + case Type_float: + return memcmp(dst, src, 4) == 0; + + case Type_int64_t: + case Type_double: + return memcmp(dst, src, 8) == 0; + + case Type_int64_t_pad: + case Type_double_pad: + return true; + + case Type_intptr_t: + return memcmp(dst, src, BytesPerWord) == 0; + + case Type_object: + return true; + + default: abort(); + } +} + +bool +nonObjectsEqual(TypeMap* map, uint8_t* src, uint8_t* dst) +{ + for (unsigned i = 0; i < map->fixedFieldCount; ++i) { + Field* field = map->fixedFields() + i; + if (not nonObjectsEqual + (src + field->offset, dst + field->targetOffset, field->type)) + { + return false; + } + } + + if (map->targetArrayElementSizeInBytes) { + unsigned fixedSize = map->buildFixedSizeInWords * BytesPerWord; + unsigned count = cast(src, fixedSize - BytesPerWord); + + for (unsigned i = 0; i < count; ++i) { + if (not nonObjectsEqual + (src + fixedSize + (i * map->buildArrayElementSizeInBytes), + dst + (map->targetFixedSizeInWords * TargetBytesPerWord) + + (i * map->targetArrayElementSizeInBytes), map->arrayElementType)) + { + return false; + } + } + } + + return true; +} + +void +copy(Thread* t, object typeMaps, object p, uint8_t* dst) +{ + TypeMap* map = typeMap(t, typeMaps, p); + + uint8_t* src = reinterpret_cast(p); + + for (unsigned i = 0; i < map->fixedFieldCount; ++i) { + Field* field = map->fixedFields() + i; + if (field->type > Type_array) abort(t); + copy(t, src + field->offset, dst + field->targetOffset, field->type); + } + + if (map->targetArrayElementSizeInBytes) { + unsigned fixedSize = map->buildFixedSizeInWords * BytesPerWord; + unsigned count = cast(p, fixedSize - BytesPerWord); + + for (unsigned i = 0; i < count; ++i) { + copy(t, src + fixedSize + (i * map->buildArrayElementSizeInBytes), + dst + (map->targetFixedSizeInWords * TargetBytesPerWord) + + (i * map->targetArrayElementSizeInBytes), map->arrayElementType); + } + } else { + switch (map->kind) { + case TypeMap::NormalKind: + break; + + case TypeMap::SingletonKind: { + uint8_t* mask = dst + (map->targetFixedSizeInWords * TargetBytesPerWord); + memset(mask, 0, singletonMaskSize + (map->targetFixedSizeInWords - 2, TargetBitsPerWord) + * TargetBytesPerWord); + + for (unsigned i = 0; i < map->fixedFieldCount; ++i) { + Field* field = map->fixedFields() + i; + if (field->type == Type_object) { + unsigned offset = field->targetOffset / TargetBytesPerWord; + reinterpret_cast(mask)[offset / 32] + |= static_cast(1) << (offset % 32); + } + } + + if (DebugNativeTarget) { + expect + (t, memcmp + (src + (map->targetFixedSizeInWords * TargetBytesPerWord), mask, + singletonMaskSize + (map->targetFixedSizeInWords - 2, TargetBitsPerWord) + * TargetBytesPerWord) == 0); + } + } break; + + case TypeMap::PoolKind: { + unsigned poolMaskSize = vm::poolMaskSize + (map->targetFixedSizeInWords - 2, TargetBitsPerWord); + + uint8_t* poolMask = dst + + (map->targetFixedSizeInWords * TargetBytesPerWord); + + memset(poolMask, 0, poolMaskSize * TargetBytesPerWord); + + uint8_t* objectMask = dst + + ((map->targetFixedSizeInWords + poolMaskSize) * TargetBytesPerWord); + + memset(objectMask, 0, singletonMaskSize + (map->targetFixedSizeInWords - 2 + poolMaskSize, + TargetBitsPerWord) * TargetBytesPerWord); + + for (unsigned i = 0; i < map->fixedFieldCount; ++i) { + Field* field = map->fixedFields() + i; + switch (field->type) { + case Type_object: + reinterpret_cast(objectMask)[i / 32] + |= static_cast(1) << (i % 32); + break; + + case Type_float: + case Type_double: + reinterpret_cast(poolMask) + [i / TargetBitsPerWord] + |= static_cast(1) << (i % TargetBitsPerWord); + break; + + default: + break; + } + } + + if (DebugNativeTarget) { + expect + (t, memcmp + (src + (map->targetFixedSizeInWords * TargetBytesPerWord), poolMask, + (poolMaskSize + singletonMaskSize + (map->targetFixedSizeInWords - 2 + poolMaskSize, + TargetBitsPerWord)) + * TargetBytesPerWord) == 0); + } + } break; + + default: abort(t); + } + } + + if (DebugNativeTarget) { + expect(t, targetSize(t, typeMaps, p) == baseSize(t, p, objectClass(t, p))); + expect(t, nonObjectsEqual(map, src, dst)); + } +} + HeapWalker* makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, - unsigned capacity, object constants) + unsigned capacity, object constants, object typeMaps) { class Visitor: public HeapVisitor { public: - Visitor(Thread* t, uintptr_t* heap, uintptr_t* map, unsigned capacity): - t(t), currentObject(0), currentNumber(0), currentOffset(0), heap(heap), - map(map), position(0), capacity(capacity) + Visitor(Thread* t, object typeMaps, uintptr_t* heap, + uintptr_t* map, unsigned capacity): + t(t), typeMaps(typeMaps), currentObject(0), currentNumber(0), + currentOffset(0), heap(heap), map(map), position(0), capacity(capacity) { } void visit(unsigned number) { if (currentObject) { - unsigned offset = currentNumber - 1 + currentOffset; + if (DebugNativeTarget) { + expect + (t, targetOffset + (t, typeMaps, currentObject, currentOffset * BytesPerWord) + == currentOffset * BytesPerWord); + } + + unsigned offset = currentNumber - 1 + + (targetOffset + (t, typeMaps, currentObject, currentOffset * BytesPerWord) + / TargetBytesPerWord); + unsigned mark = heap[offset] & (~PointerMask); unsigned value = number | (mark << BootShift); @@ -243,10 +860,11 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, virtual unsigned visitNew(object p) { if (p) { - unsigned size = objectSize(t, p); + unsigned size = targetSize(t, typeMaps, p); unsigned number; if ((currentObject + and objectClass(t, currentObject) == type(t, Machine::ClassType) and (currentOffset * BytesPerWord) == ClassStaticTable) or instanceOf(t, type(t, Machine::SystemClassLoaderType), p)) { @@ -257,24 +875,41 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, // to runtime-allocated memory would fail because we don't // scan non-fixed objects in the heap image during GC. - FixedAllocator allocator - (t->m->system, reinterpret_cast(heap + position), - (capacity - position) * BytesPerWord); + target_uintptr_t* dst = heap + position + TargetFixieSizeInWords; - unsigned totalInBytes; - uintptr_t* dst = static_cast - (t->m->heap->allocateImmortalFixed - (&allocator, size, true, &totalInBytes)); + unsigned maskSize = ceiling(size, TargetBytesPerWord); - memcpy(dst, p, size * BytesPerWord); + unsigned total = TargetFixieSizeInWords + size + maskSize; + + expect(t, position + total < capacity); + + memset(heap + position, 0, TargetFixieSizeInBytes); + + uint8_t age = FixieTenureThreshold + 1; + memcpy(reinterpret_cast(heap + position) + + TargetFixieAge, &age, 1); + + uint8_t hasMask = true; + memcpy(reinterpret_cast(heap + position) + + TargetFixieHasMask, &hasMask, 1); + + uint32_t targetSize = TARGET_V4(size); + memcpy(reinterpret_cast(heap + position) + + TargetFixieSize, &targetSize, 4); + + copy(t, typeMaps, p, reinterpret_cast(dst)); dst[0] |= FixedMark; + memset(heap + position + TargetFixieSizeInWords + size, 0, + maskSize * TargetBytesPerWord); + number = (dst - heap) + 1; - position += ceiling(totalInBytes, BytesPerWord); + position += total; } else { - assert(t, position + size < capacity); - memcpy(heap + position, p, size * BytesPerWord); + expect(t, position + size < capacity); + + copy(t, typeMaps, p, reinterpret_cast(heap + position)); number = position + 1; position += size; @@ -303,14 +938,15 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, } Thread* t; + object typeMaps; object currentObject; unsigned currentNumber; unsigned currentOffset; - uintptr_t* heap; - uintptr_t* map; + target_uintptr_t* heap; + target_uintptr_t* map; unsigned position; unsigned capacity; - } visitor(t, heap, map, capacity / BytesPerWord); + } visitor(t, typeMaps, heap, map, capacity / TargetBytesPerWord); HeapWalker* w = makeHeapWalker(t, &visitor); visitRoots(t, image, w, constants); @@ -326,7 +962,7 @@ updateConstants(Thread* t, object constants, uint8_t* code, uintptr_t* codeMap, { for (; constants; constants = tripleThird(t, constants)) { unsigned target = heapTable->find(tripleFirst(t, constants)); - assert(t, target > 0); + expect(t, target > 0); for (Promise::Listener* pl = static_cast (pointerValue(t, tripleSecond(t, constants)))->listener; @@ -334,13 +970,13 @@ updateConstants(Thread* t, object constants, uint8_t* code, uintptr_t* codeMap, { void* location; bool flat = pl->resolve(0, &location); - uintptr_t offset = target | BootHeapOffset; + target_uintptr_t offset = target | BootHeapOffset; if (flat) { offset |= BootFlatConstant; } - memcpy(location, &offset, BytesPerWord); + memcpy(location, &offset, TargetBytesPerWord); - assert(t, reinterpret_cast(location) + expect(t, reinterpret_cast(location) >= reinterpret_cast(code)); markBit(codeMap, reinterpret_cast(location) @@ -366,74 +1002,221 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, (t->m->heap->allocate(codeMapSize(codeCapacity))); memset(codeMap, 0, codeMapSize(codeCapacity)); - object constants = makeCodeImage - (t, &zone, image, code, codeMap, className, methodName, methodSpec); + object classPoolMap; + object typeMaps; + object constants; - PROTECT(t, constants); + { classPoolMap = makeHashMap(t, 0, 0); + PROTECT(t, classPoolMap); - // this map will not be used when the bootimage is loaded, so - // there's no need to preserve it: - setRoot(t, Machine::ByteArrayMap, makeWeakHashMap(t, 0, 0)); + setRoot(t, Machine::PoolMap, classPoolMap); - // name all primitive classes so we don't try to update immutable - // references at runtime: - { object name = makeByteArray(t, "void"); - set(t, type(t, Machine::JvoidType), ClassName, name); + typeMaps = makeHashMap(t, 0, 0); + PROTECT(t, typeMaps); + + constants = makeCodeImage + (t, &zone, image, code, codeMap, className, methodName, methodSpec, + typeMaps); + + PROTECT(t, constants); + +#include "type-maps.cpp" + + for (unsigned i = 0; i < arrayLength(t, t->m->types); ++i) { + Type* source = types[i]; + unsigned count = 0; + while (source[count] != Type_none) { + ++ count; + } + ++ count; + + Type types[count]; + types[0] = Type_object; + unsigned buildOffsets[count]; + buildOffsets[0] = 0; + unsigned buildOffset = BytesPerWord; + unsigned targetOffsets[count]; + targetOffsets[0] = 0; + unsigned targetOffset = TargetBytesPerWord; + bool sawArray = false; + unsigned buildSize = BytesPerWord; + unsigned targetSize = TargetBytesPerWord; + for (unsigned j = 1; j < count; ++j) { + switch (source[j - 1]) { + case Type_object: + types[j] = Type_object; + buildSize = BytesPerWord; + targetSize = TargetBytesPerWord; + break; + + case Type_word: + case Type_intptr_t: + case Type_uintptr_t: + types[j] = Type_intptr_t; + buildSize = BytesPerWord; + targetSize = TargetBytesPerWord; + break; + + case Type_int8_t: + case Type_uint8_t: + types[j] = Type_int8_t; + buildSize = targetSize = 1; + break; + + case Type_int16_t: + case Type_uint16_t: + types[j] = Type_int16_t; + buildSize = targetSize = 2; + break; + + case Type_int32_t: + case Type_uint32_t: + case Type_float: + types[j] = Type_int32_t; + buildSize = targetSize = 4; + break; + + case Type_int64_t: + case Type_uint64_t: + case Type_double: + types[j] = Type_int64_t; + buildSize = targetSize = 8; + break; + + case Type_array: + types[j] = Type_none; + buildSize = targetSize = 0; + break; + + default: abort(t); + } + + if (source[j - 1] == Type_array) { + sawArray = true; + } + + if (not sawArray) { + while (buildOffset % buildSize) { + ++ buildOffset; + } + + buildOffsets[j] = buildOffset; + + buildOffset += buildSize; + + while (targetOffset % targetSize) { + ++ targetOffset; + } + + targetOffsets[j] = targetOffset; + + targetOffset += targetSize; + } + } + + unsigned fixedFieldCount; + Type arrayElementType; + unsigned buildArrayElementSize; + unsigned targetArrayElementSize; + if (sawArray) { + fixedFieldCount = count - 2; + arrayElementType = types[count - 1]; + buildArrayElementSize = buildSize; + targetArrayElementSize = targetSize; + } else { + fixedFieldCount = count; + arrayElementType = Type_none; + buildArrayElementSize = 0; + targetArrayElementSize = 0; + } + + object array = makeByteArray + (t, TypeMap::sizeInBytes + (ceiling(buildOffset, BytesPerWord), fixedFieldCount)); + + TypeMap* map = new (&byteArrayBody(t, array, 0)) TypeMap + (ceiling(buildOffset, BytesPerWord), + ceiling(targetOffset, TargetBytesPerWord), + fixedFieldCount, TypeMap::NormalKind, buildArrayElementSize, + targetArrayElementSize, arrayElementType); + + for (unsigned j = 0; j < fixedFieldCount; ++j) { + expect(t, buildOffsets[j] < map->buildFixedSizeInWords * BytesPerWord); + + map->targetFixedOffsets()[buildOffsets[j]] = targetOffsets[j]; + + new (map->fixedFields() + j) Field + (types[j], buildOffsets[j], targetOffsets[j]); + } + + hashMapInsertOrReplace + (t, typeMaps, type(t, static_cast(i)), array, + objectHash, objectEqual); + } + + // these roots will not be used when the bootimage is loaded, so + // there's no need to preserve them: + setRoot(t, Machine::PoolMap, 0); + setRoot(t, Machine::ByteArrayMap, makeWeakHashMap(t, 0, 0)); + + // name all primitive classes so we don't try to update immutable + // references at runtime: + { object name = makeByteArray(t, "void"); + set(t, type(t, Machine::JvoidType), ClassName, name); - name = makeByteArray(t, "boolean"); - set(t, type(t, Machine::JbooleanType), ClassName, name); + name = makeByteArray(t, "boolean"); + set(t, type(t, Machine::JbooleanType), ClassName, name); - name = makeByteArray(t, "byte"); - set(t, type(t, Machine::JbyteType), ClassName, name); + name = makeByteArray(t, "byte"); + set(t, type(t, Machine::JbyteType), ClassName, name); - name = makeByteArray(t, "short"); - set(t, type(t, Machine::JshortType), ClassName, name); + name = makeByteArray(t, "short"); + set(t, type(t, Machine::JshortType), ClassName, name); - name = makeByteArray(t, "char"); - set(t, type(t, Machine::JcharType), ClassName, name); + name = makeByteArray(t, "char"); + set(t, type(t, Machine::JcharType), ClassName, name); - name = makeByteArray(t, "int"); - set(t, type(t, Machine::JintType), ClassName, name); + name = makeByteArray(t, "int"); + set(t, type(t, Machine::JintType), ClassName, name); - name = makeByteArray(t, "float"); - set(t, type(t, Machine::JfloatType), ClassName, name); + name = makeByteArray(t, "float"); + set(t, type(t, Machine::JfloatType), ClassName, name); - name = makeByteArray(t, "long"); - set(t, type(t, Machine::JlongType), ClassName, name); + name = makeByteArray(t, "long"); + set(t, type(t, Machine::JlongType), ClassName, name); - name = makeByteArray(t, "double"); - set(t, type(t, Machine::JdoubleType), ClassName, name); + name = makeByteArray(t, "double"); + set(t, type(t, Machine::JdoubleType), ClassName, name); + } + + // resolve primitive array classes in case they are needed at + // runtime: + { object name = makeByteArray(t, "[B"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[Z"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[S"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[C"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[I"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[J"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[F"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + + name = makeByteArray(t, "[D"); + resolveSystemClass(t, root(t, Machine::BootLoader), name, true); + } } - // resolve primitive array classes in case they are needed at - // runtime: - { object name = makeByteArray(t, "[B"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[Z"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[S"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[C"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[I"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[J"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[F"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - - name = makeByteArray(t, "[D"); - resolveSystemClass(t, root(t, Machine::BootLoader), name, true); - } - - collect(t, Heap::MajorCollection); - uintptr_t* heap = static_cast (t->m->heap->allocate(HeapCapacity)); uintptr_t* heapMap = static_cast @@ -441,7 +1224,7 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, memset(heapMap, 0, heapMapSize(HeapCapacity)); HeapWalker* heapWalker = makeHeapImage - (t, image, heap, heapMap, HeapCapacity, constants); + (t, image, heap, heapMap, HeapCapacity, constants, typeMaps); updateConstants(t, constants, code, codeMap, heapWalker->map()); @@ -512,7 +1295,7 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, + (image->stringCount * sizeof(unsigned)) + (image->callCount * sizeof(unsigned) * 2); - while (offset % BytesPerWord) { + while (offset % TargetBytesPerWord) { uint8_t c = 0; fwrite(&c, 1, 1, out); ++ offset; diff --git a/src/compile.cpp b/src/compile.cpp index afd590fc8f..fa1ffa4778 100644 --- a/src/compile.cpp +++ b/src/compile.cpp @@ -13,6 +13,7 @@ #include "vector.h" #include "process.h" #include "assembler.h" +#include "target.h" #include "compiler.h" #include "arch.h" @@ -52,7 +53,7 @@ const bool Continuations = true; const bool Continuations = false; #endif -const unsigned MaxNativeCallFootprint = BytesPerWord == 8 ? 4 : 5; +const unsigned MaxNativeCallFootprint = TargetBytesPerWord == 8 ? 4 : 5; const unsigned InitialZoneCapacityInBytes = 64 * 1024; @@ -3164,7 +3165,7 @@ resultSize(MyThread* t, unsigned code) return 4; case ObjectField: - return BytesPerWord; + return TargetBytesPerWord; case LongField: case DoubleField: @@ -3305,8 +3306,8 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall, methodParameterFootprint(t, target)); c->store - (BytesPerWord, frame->addressOperand(returnAddressPromise), - BytesPerWord, c->memory + (TargetBytesPerWord, frame->addressOperand(returnAddressPromise), + TargetBytesPerWord, c->memory (c->register_(t->arch->thread()), Compiler::AddressType, difference(&(t->tailAddress), t))); @@ -3455,7 +3456,7 @@ compileDirectReferenceInvoke(MyThread* t, Frame* frame, Thunk thunk, (c->constant(getThunk(t, thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::AddressType, 2, c->register_(t->arch->thread()), frame->append(pair)), reference, isStatic, tailCall); @@ -3497,7 +3498,7 @@ compileDirectAbstractInvoke(MyThread* t, Frame* frame, Thunk thunk, (c->constant(getThunk(t, thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::AddressType, 2, c->register_(t->arch->thread()), frame->append(target)), target, tailCall); @@ -3560,7 +3561,7 @@ inTryBlock(MyThread* t, object code, unsigned ip) if (table) { unsigned length = exceptionHandlerTableLength(t, table); for (unsigned i = 0; i < length; ++i) { - ExceptionHandler* eh = exceptionHandlerTableBody(t, table, i); + uint64_t eh = exceptionHandlerTableBody(t, table, i); if (ip >= exceptionHandlerStart(eh) and ip < exceptionHandlerEnd(eh)) { @@ -3867,16 +3868,18 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, case aaload: frame->pushObject (c->load - (BytesPerWord, BytesPerWord, c->memory - (array, Compiler::ObjectType, ArrayBody, index, BytesPerWord), - BytesPerWord)); + (TargetBytesPerWord, TargetBytesPerWord, c->memory + (array, Compiler::ObjectType, ArrayBody, index, + TargetBytesPerWord), + TargetBytesPerWord)); break; case faload: frame->pushInt (c->load (4, 4, c->memory - (array, Compiler::FloatType, ArrayBody, index, 4), BytesPerWord)); + (array, Compiler::FloatType, ArrayBody, index, 4), + TargetBytesPerWord)); break; case iaload: @@ -3884,7 +3887,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (4, 4, c->memory (array, Compiler::IntegerType, ArrayBody, index, 4), - BytesPerWord)); + TargetBytesPerWord)); break; case baload: @@ -3892,7 +3895,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (1, 1, c->memory (array, Compiler::IntegerType, ArrayBody, index, 1), - BytesPerWord)); + TargetBytesPerWord)); break; case caload: @@ -3900,7 +3903,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->loadz (2, 2, c->memory (array, Compiler::IntegerType, ArrayBody, index, 2), - BytesPerWord)); + TargetBytesPerWord)); break; case daload: @@ -3922,7 +3925,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (2, 2, c->memory (array, Compiler::IntegerType, ArrayBody, index, 2), - BytesPerWord)); + TargetBytesPerWord)); break; } } break; @@ -3968,32 +3971,33 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, c->add (4, c->constant(ArrayBody, Compiler::IntegerType), c->shl - (4, c->constant(log(BytesPerWord), Compiler::IntegerType), index)), + (4, c->constant(log(TargetBytesPerWord), Compiler::IntegerType), + index)), value); } break; case fastore: c->store - (BytesPerWord, value, 4, c->memory + (TargetBytesPerWord, value, 4, c->memory (array, Compiler::FloatType, ArrayBody, index, 4)); break; case iastore: c->store - (BytesPerWord, value, 4, c->memory + (TargetBytesPerWord, value, 4, c->memory (array, Compiler::IntegerType, ArrayBody, index, 4)); break; case bastore: c->store - (BytesPerWord, value, 1, c->memory + (TargetBytesPerWord, value, 1, c->memory (array, Compiler::IntegerType, ArrayBody, index, 1)); break; case castore: case sastore: c->store - (BytesPerWord, value, 2, c->memory + (TargetBytesPerWord, value, 2, c->memory (array, Compiler::IntegerType, ArrayBody, index, 2)); break; @@ -4062,7 +4066,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->constant(getThunk(t, thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::ObjectType, 3, c->register_(t->arch->thread()), frame->append(argument), length)); @@ -4070,16 +4074,16 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, case areturn: { handleExit(t, frame); - c->return_(BytesPerWord, frame->popObject()); + c->return_(TargetBytesPerWord, frame->popObject()); } return; case arraylength: { frame->pushInt (c->load - (BytesPerWord, BytesPerWord, + (TargetBytesPerWord, TargetBytesPerWord, c->memory (frame->popObject(), Compiler::IntegerType, ArrayLength, 0, 1), - BytesPerWord)); + TargetBytesPerWord)); } break; case astore: @@ -4367,7 +4371,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, if (LIKELY(field)) { if ((fieldFlags(t, field) & ACC_VOLATILE) - and BytesPerWord == 4 + and TargetBytesPerWord == 4 and (fieldCode(t, field) == DoubleField or fieldCode(t, field) == LongField)) { @@ -4375,7 +4379,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, c->call (c->constant - (getThunk(t, acquireMonitorForObjectThunk), Compiler::AddressType), + (getThunk(t, acquireMonitorForObjectThunk), + Compiler::AddressType), 0, frame->trace(0, 0), 0, Compiler::VoidType, 2, c->register_(t->arch->thread()), frame->append(field)); @@ -4421,7 +4426,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (1, 1, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - BytesPerWord)); + TargetBytesPerWord)); break; case CharField: @@ -4429,7 +4434,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->loadz (2, 2, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - BytesPerWord)); + TargetBytesPerWord)); break; case ShortField: @@ -4437,7 +4442,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (2, 2, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - BytesPerWord)); + TargetBytesPerWord)); break; case FloatField: @@ -4445,7 +4450,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (4, 4, c->memory (table, Compiler::FloatType, fieldOffset(t, field), 0, 1), - BytesPerWord)); + TargetBytesPerWord)); break; case IntField: @@ -4453,7 +4458,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (4, 4, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - BytesPerWord)); + TargetBytesPerWord)); break; case DoubleField: @@ -4473,10 +4478,10 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, case ObjectField: frame->pushObject (c->load - (BytesPerWord, BytesPerWord, + (TargetBytesPerWord, TargetBytesPerWord, c->memory (table, Compiler::ObjectType, fieldOffset(t, field), 0, 1), - BytesPerWord)); + TargetBytesPerWord)); break; default: @@ -4484,7 +4489,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } if (fieldFlags(t, field) & ACC_VOLATILE) { - if (BytesPerWord == 4 + if (TargetBytesPerWord == 4 and (fieldCode(t, field) == DoubleField or fieldCode(t, field) == LongField)) { @@ -4551,11 +4556,13 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } break; case i2b: { - frame->pushInt(c->load(BytesPerWord, 1, frame->popInt(), BytesPerWord)); + frame->pushInt + (c->load(TargetBytesPerWord, 1, frame->popInt(), TargetBytesPerWord)); } break; case i2c: { - frame->pushInt(c->loadz(BytesPerWord, 2, frame->popInt(), BytesPerWord)); + frame->pushInt + (c->loadz(TargetBytesPerWord, 2, frame->popInt(), TargetBytesPerWord)); } break; case i2d: { @@ -4567,11 +4574,12 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } break; case i2l: - frame->pushLong(c->load(BytesPerWord, 4, frame->popInt(), 8)); + frame->pushLong(c->load(TargetBytesPerWord, 4, frame->popInt(), 8)); break; case i2s: { - frame->pushInt(c->load(BytesPerWord, 2, frame->popInt(), BytesPerWord)); + frame->pushInt + (c->load(TargetBytesPerWord, 2, frame->popInt(), TargetBytesPerWord)); } break; case iadd: { @@ -4637,9 +4645,9 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, Compiler::Operand* target = frame->machineIp(newIp); if (instruction == if_acmpeq) { - c->jumpIfEqual(BytesPerWord, a, b, target); + c->jumpIfEqual(TargetBytesPerWord, a, b, target); } else { - c->jumpIfNotEqual(BytesPerWord, a, b, target); + c->jumpIfNotEqual(TargetBytesPerWord, a, b, target); } saveStateAndCompile(t, frame, newIp); @@ -4737,9 +4745,9 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, Compiler::Operand* target = frame->machineIp(newIp); if (instruction == ifnull) { - c->jumpIfEqual(BytesPerWord, a, b, target); + c->jumpIfEqual(TargetBytesPerWord, a, b, target); } else { - c->jumpIfNotEqual(BytesPerWord, a, b, target); + c->jumpIfNotEqual(TargetBytesPerWord, a, b, target); } saveStateAndCompile(t, frame, newIp); @@ -4868,7 +4876,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->constant(getThunk(t, thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::AddressType, 3, c->register_(t->arch->thread()), frame->append(argument), c->peek(1, parameterFootprint - 1)), @@ -4976,7 +4984,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, Compiler::Operand* result = c->stackCall (c->memory (c->and_ - (BytesPerWord, c->constant(PointerMask, Compiler::IntegerType), + (TargetBytesPerWord, c->constant + (PointerMask, Compiler::IntegerType), c->memory(instance, Compiler::ObjectType, 0, 0, 1)), Compiler::ObjectType, offset, 0, 1), tailCall ? Compiler::TailJump : 0, @@ -5008,7 +5017,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::AddressType, 3, c->register_(t->arch->thread()), frame->append(pair), c->peek(1, methodReferenceParameterFootprint @@ -5132,7 +5141,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } break; case l2i: - frame->pushInt(c->load(8, 8, frame->popLong(), BytesPerWord)); + frame->pushInt(c->load(8, 8, frame->popLong(), TargetBytesPerWord)); break; case ladd: { @@ -5201,7 +5210,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::ObjectType, 2, c->register_(t->arch->thread()), frame->append(makePair(t, context->method, reference)))); @@ -5216,7 +5225,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (getThunk(t, getJClass64Thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::ObjectType, 2, c->register_(t->arch->thread()), frame->append(v))); } else { @@ -5330,7 +5339,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->call (c->constant (getThunk(t, lookUpAddressThunk), Compiler::AddressType), - 0, 0, BytesPerWord, Compiler::AddressType, + 0, 0, TargetBytesPerWord, Compiler::AddressType, 4, key, start, c->constant(pairCount, Compiler::IntegerType), default_)); @@ -5477,7 +5486,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (getThunk(t, thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::ObjectType, 4, c->register_(t->arch->thread()), frame->append(argument), c->constant(dimensions, Compiler::IntegerType), @@ -5516,7 +5525,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->constant(getThunk(t, thunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::ObjectType, 2, c->register_(t->arch->thread()), frame->append(argument))); } break; @@ -5531,7 +5540,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->constant(getThunk(t, makeBlankArrayThunk), Compiler::AddressType), 0, frame->trace(0, 0), - BytesPerWord, + TargetBytesPerWord, Compiler::ObjectType, 3, c->register_(t->arch->thread()), c->constant(type, Compiler::IntegerType), length)); @@ -5593,7 +5602,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } if (fieldFlags(t, field) & ACC_VOLATILE) { - if (BytesPerWord == 4 + if (TargetBytesPerWord == 4 and (fieldCode == DoubleField or fieldCode == LongField)) { PROTECT(t, field); @@ -5625,26 +5634,26 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, case ByteField: case BooleanField: c->store - (BytesPerWord, value, 1, c->memory + (TargetBytesPerWord, value, 1, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); break; case CharField: case ShortField: c->store - (BytesPerWord, value, 2, c->memory + (TargetBytesPerWord, value, 2, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); break; case FloatField: c->store - (BytesPerWord, value, 4, c->memory + (TargetBytesPerWord, value, 4, c->memory (table, Compiler::FloatType, fieldOffset(t, field), 0, 1)); break; case IntField: c->store - (BytesPerWord, value, 4, c->memory + (TargetBytesPerWord, value, 4, c->memory (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); break; @@ -5686,7 +5695,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } if (fieldFlags(t, field) & ACC_VOLATILE) { - if (BytesPerWord == 4 + if (TargetBytesPerWord == 4 and (fieldCode == DoubleField or fieldCode == LongField)) { c->call @@ -5862,9 +5871,9 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, c->jmp (c->load - (BytesPerWord, BytesPerWord, c->memory - (start, Compiler::AddressType, 0, normalizedKey, BytesPerWord), - BytesPerWord)); + (TargetBytesPerWord, TargetBytesPerWord, c->memory + (start, Compiler::AddressType, 0, normalizedKey, TargetBytesPerWord), + TargetBytesPerWord)); Compiler::State* state = c->saveState(); @@ -6018,9 +6027,9 @@ truncateLineNumberTable(Thread* t, object table, unsigned length) PROTECT(t, table); object newTable = makeLineNumberTable(t, length); - memcpy(lineNumberTableBody(t, newTable, 0), - lineNumberTableBody(t, table, 0), - length * sizeof(LineNumber)); + memcpy(&lineNumberTableBody(t, newTable, 0), + &lineNumberTableBody(t, table, 0), + length * sizeof(uint64_t)); return newTable; } @@ -6046,7 +6055,7 @@ translateExceptionHandlerTable(MyThread* t, Context* context, intptr_t start) unsigned ni = 0; for (unsigned oi = 0; oi < length; ++ oi) { - ExceptionHandler* oldHandler = exceptionHandlerTableBody + uint64_t oldHandler = exceptionHandlerTableBody (t, oldTable, oi); int handlerStart = resolveIpForwards @@ -6107,8 +6116,7 @@ translateLineNumberTable(MyThread* t, Context* context, intptr_t start) object newTable = makeLineNumberTable(t, length); unsigned ni = 0; for (unsigned oi = 0; oi < length; ++oi) { - LineNumber* oldLine = lineNumberTableBody(t, oldTable, oi); - LineNumber* newLine = lineNumberTableBody(t, newTable, ni); + uint64_t oldLine = lineNumberTableBody(t, oldTable, oi); int ip = resolveIpForwards (context, lineNumberIp(oldLine), oi + 1 < length @@ -6116,12 +6124,9 @@ translateLineNumberTable(MyThread* t, Context* context, intptr_t start) : lineNumberIp(oldLine) + 1); if (LIKELY(ip >= 0)) { - lineNumberIp(newLine) - = context->compiler->machineIp(ip)->value() - start; - - lineNumberLine(newLine) = lineNumberLine(oldLine); - - ++ ni; + lineNumberTableBody(t, newTable, ni++) = lineNumber + (context->compiler->machineIp(ip)->value() - start, + lineNumberLine(oldLine)); } } @@ -6675,7 +6680,9 @@ makeGeneralFrameMapTable(MyThread* t, Context* context, uint8_t* start, pathIndex = subroutine->tableIndex; - THREAD_RUNTIME_ARRAY(t, SubroutineTrace*, traces, p->subroutineTraceCount); + THREAD_RUNTIME_ARRAY + (t, SubroutineTrace*, traces, p->subroutineTraceCount); + unsigned i = 0; for (SubroutineTrace* trace = p->subroutineTrace; trace; trace = trace->next) @@ -6795,9 +6802,9 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) // we must acquire the class lock here at the latest unsigned codeSize = c->resolve - (allocator->base + allocator->offset + BytesPerWord); + (allocator->base + allocator->offset + TargetBytesPerWord); - unsigned total = pad(codeSize) + pad(c->poolSize()) + BytesPerWord; + unsigned total = pad(codeSize) + pad(c->poolSize()) + TargetBytesPerWord; uintptr_t* code = static_cast(allocator->allocate(total)); code[0] = codeSize; @@ -6810,7 +6817,8 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) if (context->objectPool) { object pool = allocate3 (t, allocator, Machine::ImmortalAllocation, - FixedSizeOfArray + ((context->objectPoolCount + 1) * BytesPerWord), + FixedSizeOfArray + + ((context->objectPoolCount + 1) * TargetBytesPerWord), true); initArray(t, pool, context->objectPoolCount + 1); @@ -6821,7 +6829,7 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) unsigned i = 1; for (PoolElement* p = context->objectPool; p; p = p->next) { - unsigned offset = ArrayBody + ((i++) * BytesPerWord); + unsigned offset = ArrayBody + ((i++) * TargetBytesPerWord); p->address = reinterpret_cast(pool) + offset; @@ -7023,7 +7031,7 @@ compile(MyThread* t, Context* context) progress = false; for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) { - ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i); + uint64_t eh = exceptionHandlerTableBody(t, eht, i); int start = resolveIpForwards (context, exceptionHandlerStart(eh), exceptionHandlerEnd(eh)); @@ -7042,7 +7050,7 @@ compile(MyThread* t, Context* context) Frame frame2(&frame, RUNTIME_ARRAY_BODY(stackMap)); unsigned end = exceptionHandlerEnd(eh); - if (exceptionHandlerIp(eh) >= start + if (exceptionHandlerIp(eh) >= static_cast(start) and exceptionHandlerIp(eh) < end) { end = exceptionHandlerIp(eh); @@ -9334,15 +9342,15 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) p->thunks.default_.frameSavedOffset = a->length(); Assembler::Register thread(t->arch->thread()); - a->pushFrame(1, BytesPerWord, RegisterOperand, &thread); + a->pushFrame(1, TargetBytesPerWord, RegisterOperand, &thread); Assembler::Constant proc(&(defaultContext.promise)); - a->apply(LongCall, BytesPerWord, ConstantOperand, &proc); + a->apply(LongCall, TargetBytesPerWord, ConstantOperand, &proc); a->popFrame(t->arch->alignFrameSize(1)); Assembler::Register result(t->arch->returnLow()); - a->apply(Jump, BytesPerWord, RegisterOperand, &result); + a->apply(Jump, TargetBytesPerWord, RegisterOperand, &result); p->thunks.default_.length = a->endBlock(false)->resolve(0, 0); } @@ -9355,38 +9363,38 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) Assembler::Memory virtualCallTargetSrc (t->arch->stack(), (t->arch->frameFooterSize() + t->arch->frameReturnAddressSize()) - * BytesPerWord); + * TargetBytesPerWord); - a->apply(Move, BytesPerWord, MemoryOperand, &virtualCallTargetSrc, - BytesPerWord, RegisterOperand, &class_); + a->apply(Move, TargetBytesPerWord, MemoryOperand, &virtualCallTargetSrc, + TargetBytesPerWord, RegisterOperand, &class_); Assembler::Memory virtualCallTargetDst (t->arch->thread(), difference(&(t->virtualCallTarget), t)); - a->apply(Move, BytesPerWord, RegisterOperand, &class_, - BytesPerWord, MemoryOperand, &virtualCallTargetDst); + a->apply(Move, TargetBytesPerWord, RegisterOperand, &class_, + TargetBytesPerWord, MemoryOperand, &virtualCallTargetDst); Assembler::Register index(t->arch->virtualCallIndex()); Assembler::Memory virtualCallIndex (t->arch->thread(), difference(&(t->virtualCallIndex), t)); - a->apply(Move, BytesPerWord, RegisterOperand, &index, - BytesPerWord, MemoryOperand, &virtualCallIndex); + a->apply(Move, TargetBytesPerWord, RegisterOperand, &index, + TargetBytesPerWord, MemoryOperand, &virtualCallIndex); a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); p->thunks.defaultVirtual.frameSavedOffset = a->length(); Assembler::Register thread(t->arch->thread()); - a->pushFrame(1, BytesPerWord, RegisterOperand, &thread); + a->pushFrame(1, TargetBytesPerWord, RegisterOperand, &thread); Assembler::Constant proc(&(defaultVirtualContext.promise)); - a->apply(LongCall, BytesPerWord, ConstantOperand, &proc); + a->apply(LongCall, TargetBytesPerWord, ConstantOperand, &proc); a->popFrame(t->arch->alignFrameSize(1)); Assembler::Register result(t->arch->returnLow()); - a->apply(Jump, BytesPerWord, RegisterOperand, &result); + a->apply(Jump, TargetBytesPerWord, RegisterOperand, &result); p->thunks.defaultVirtual.length = a->endBlock(false)->resolve(0, 0); } @@ -9400,10 +9408,10 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) p->thunks.native.frameSavedOffset = a->length(); Assembler::Register thread(t->arch->thread()); - a->pushFrame(1, BytesPerWord, RegisterOperand, &thread); + a->pushFrame(1, TargetBytesPerWord, RegisterOperand, &thread); Assembler::Constant proc(&(nativeContext.promise)); - a->apply(LongCall, BytesPerWord, ConstantOperand, &proc); + a->apply(LongCall, TargetBytesPerWord, ConstantOperand, &proc); a->popFrameAndUpdateStackAndReturn (t->arch->alignFrameSize(1), difference(&(t->stack), t)); @@ -9420,10 +9428,10 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) p->thunks.aioob.frameSavedOffset = a->length(); Assembler::Register thread(t->arch->thread()); - a->pushFrame(1, BytesPerWord, RegisterOperand, &thread); + a->pushFrame(1, TargetBytesPerWord, RegisterOperand, &thread); Assembler::Constant proc(&(aioobContext.promise)); - a->apply(LongCall, BytesPerWord, ConstantOperand, &proc); + a->apply(LongCall, TargetBytesPerWord, ConstantOperand, &proc); p->thunks.aioob.length = a->endBlock(false)->resolve(0, 0); } @@ -9437,10 +9445,10 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) p->thunks.stackOverflow.frameSavedOffset = a->length(); Assembler::Register thread(t->arch->thread()); - a->pushFrame(1, BytesPerWord, RegisterOperand, &thread); + a->pushFrame(1, TargetBytesPerWord, RegisterOperand, &thread); Assembler::Constant proc(&(stackOverflowContext.promise)); - a->apply(LongCall, BytesPerWord, ConstantOperand, &proc); + a->apply(LongCall, TargetBytesPerWord, ConstantOperand, &proc); p->thunks.stackOverflow.length = a->endBlock(false)->resolve(0, 0); } @@ -9454,7 +9462,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) p->thunks.table.frameSavedOffset = a->length(); Assembler::Constant proc(&(tableContext.promise)); - a->apply(LongJump, BytesPerWord, ConstantOperand, &proc); + a->apply(LongJump, TargetBytesPerWord, ConstantOperand, &proc); p->thunks.table.length = a->endBlock(false)->resolve(0, 0); } @@ -9633,12 +9641,12 @@ compileVirtualThunk(MyThread* t, unsigned index, unsigned* size) ResolvedPromise indexPromise(index); Assembler::Constant indexConstant(&indexPromise); Assembler::Register indexRegister(t->arch->virtualCallIndex()); - a->apply(Move, BytesPerWord, ConstantOperand, &indexConstant, - BytesPerWord, RegisterOperand, &indexRegister); + a->apply(Move, TargetBytesPerWord, ConstantOperand, &indexConstant, + TargetBytesPerWord, RegisterOperand, &indexRegister); ResolvedPromise defaultVirtualThunkPromise(defaultVirtualThunk(t)); Assembler::Constant thunk(&defaultVirtualThunkPromise); - a->apply(Jump, BytesPerWord, ConstantOperand, &thunk); + a->apply(Jump, TargetBytesPerWord, ConstantOperand, &thunk); *size = a->endBlock(false)->resolve(0, 0); @@ -9723,7 +9731,7 @@ compile(MyThread* t, FixedAllocator* allocator, BootContext* bootContext, // resolve all exception handler catch types before we acquire // the class lock: for (unsigned i = 0; i < exceptionHandlerTableLength(t, ehTable); ++i) { - ExceptionHandler* handler = exceptionHandlerTableBody(t, ehTable, i); + uint64_t handler = exceptionHandlerTableBody(t, ehTable, i); if (exceptionHandlerCatchType(handler)) { resolveClassInPool (t, clone, exceptionHandlerCatchType(handler) - 1); diff --git a/src/compiler.cpp b/src/compiler.cpp index 0b7e8d1917..157bb82d61 100644 --- a/src/compiler.cpp +++ b/src/compiler.cpp @@ -10,6 +10,7 @@ #include "compiler.h" #include "assembler.h" +#include "target.h" using namespace vm; @@ -34,7 +35,7 @@ const unsigned StealRegisterReserveCount = 2; // this should be equal to the largest number of registers used by a // compare instruction: -const unsigned ResolveRegisterReserveCount = (BytesPerWord == 8 ? 2 : 4); +const unsigned ResolveRegisterReserveCount = (TargetBytesPerWord == 8 ? 2 : 4); const unsigned RegisterCopyCost = 1; const unsigned AddressCopyCost = 2; @@ -143,7 +144,7 @@ class Site { virtual SiteMask nextWordMask(Context*, unsigned) = 0; - virtual unsigned registerSize(Context*) { return BytesPerWord; } + virtual unsigned registerSize(Context*) { return TargetBytesPerWord; } virtual unsigned registerMask(Context*) { return 0; } @@ -467,8 +468,9 @@ class PoolPromise: public Promise { virtual int64_t value() { if (resolved()) { - return reinterpret_cast - (c->machineCode + pad(c->machineCodeSize) + (key * BytesPerWord)); + return reinterpret_cast + (c->machineCode + pad(c->machineCodeSize) + + (key * TargetBytesPerWord)); } abort(c); @@ -740,18 +742,18 @@ frameIndexToOffset(Context* c, unsigned frameIndex) { assert(c, frameIndex < totalFrameSize(c)); - return (frameIndex + c->arch->frameFooterSize()) * BytesPerWord; + return (frameIndex + c->arch->frameFooterSize()) * TargetBytesPerWord; } unsigned offsetToFrameIndex(Context* c, unsigned offset) { assert(c, static_cast - ((offset / BytesPerWord) - c->arch->frameFooterSize()) >= 0); - assert(c, ((offset / BytesPerWord) - c->arch->frameFooterSize()) + ((offset / TargetBytesPerWord) - c->arch->frameFooterSize()) >= 0); + assert(c, ((offset / TargetBytesPerWord) - c->arch->frameFooterSize()) < totalFrameSize(c)); - return (offset / BytesPerWord) - c->arch->frameFooterSize(); + return (offset / TargetBytesPerWord) - c->arch->frameFooterSize(); } unsigned @@ -837,7 +839,7 @@ class SiteIterator { Site** findNext(Site** p) { while (true) { if (*p) { - if (pass == 0 or (*p)->registerSize(c) > BytesPerWord) { + if (pass == 0 or (*p)->registerSize(c) > TargetBytesPerWord) { return p; } else { p = &((*p)->next); @@ -922,7 +924,7 @@ uniqueSite(Context* c, Value* v, Site* s) if (it.hasMore()) { // the site is not this word's only site, but if the site is // shared with the next word, it may be that word's only site - if (v->nextWord != v and s->registerSize(c) > BytesPerWord) { + if (v->nextWord != v and s->registerSize(c) > TargetBytesPerWord) { SiteIterator nit(c, v->nextWord); Site* p = nit.next(); if (nit.hasMore()) { @@ -1052,7 +1054,7 @@ deadWord(Context* c, Value* v) for (SiteIterator it(c, v, true, false); it.hasMore();) { Site* s = it.next(); - if (s->registerSize(c) > BytesPerWord) { + if (s->registerSize(c) > TargetBytesPerWord) { it.remove(c); addSite(c, nextWord, s); } @@ -1838,7 +1840,7 @@ class RegisterSite: public Site { RegisterSite* rs = static_cast(s); unsigned size = rs->registerSize(c); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { assert(c, number != NoRegister); return number == rs->number; } else { @@ -1940,7 +1942,7 @@ class RegisterSite: public Site { virtual SiteMask nextWordMask(Context* c, unsigned) { assert(c, number != NoRegister); - if (registerSize(c) > BytesPerWord) { + if (registerSize(c) > TargetBytesPerWord) { return SiteMask (1 << RegisterOperand, number, NoFrameIndex); } else { @@ -1955,7 +1957,7 @@ class RegisterSite: public Site { if ((1 << number) & c->arch->floatRegisterMask()) { return c->arch->floatRegisterSize(); } else { - return BytesPerWord; + return TargetBytesPerWord; } } @@ -2065,9 +2067,9 @@ class MemorySite: public Site { MemorySite* ms = static_cast(s); return ms->base == this->base and ((index == 1 and ms->offset == static_cast - (this->offset + BytesPerWord)) + (this->offset + TargetBytesPerWord)) or (index == 0 and this->offset == static_cast - (ms->offset + BytesPerWord))) + (ms->offset + TargetBytesPerWord))) and ms->index == this->index and ms->scale == this->scale; } else { @@ -2149,7 +2151,7 @@ class MemorySite: public Site { assert(c, high == this or (static_cast(high)->base == base and static_cast(high)->offset - == static_cast(offset + BytesPerWord) + == static_cast(offset + TargetBytesPerWord) and static_cast(high)->index == index and static_cast(high)->scale == scale)); @@ -2164,7 +2166,7 @@ class MemorySite: public Site { Site* copyHalf(Context* c, bool add) { if (add) { - return memorySite(c, base, offset + BytesPerWord, index, scale); + return memorySite(c, base, offset + TargetBytesPerWord, index, scale); } else { return copy(c); } @@ -2181,7 +2183,7 @@ class MemorySite: public Site { virtual Site* makeNextWord(Context* c, unsigned index) { return memorySite (c, base, offset + ((index == 1) xor c->arch->bigEndian() - ? BytesPerWord : -BytesPerWord), + ? TargetBytesPerWord : -TargetBytesPerWord), this->index, scale); } @@ -2398,7 +2400,7 @@ maybeMove(Context* c, Read* read, bool intersectRead, bool includeNextWord, unsigned registerReserveCount = 0) { Value* value = read->value; - unsigned size = value == value->nextWord ? BytesPerWord : 8; + unsigned size = value == value->nextWord ? TargetBytesPerWord : 8; class MyCostCalculator: public CostCalculator { public: @@ -2926,8 +2928,8 @@ move(Context* c, Value* value, Site* src, Site* dst) unsigned srcSize; unsigned dstSize; if (value->nextWord == value) { - srcSize = BytesPerWord; - dstSize = BytesPerWord; + srcSize = TargetBytesPerWord; + dstSize = TargetBytesPerWord; } else { srcSize = src->registerSize(c); dstSize = dst->registerSize(c); @@ -2935,7 +2937,7 @@ move(Context* c, Value* value, Site* src, Site* dst) if (srcSize == dstSize) { apply(c, Move, srcSize, src, src, dstSize, dst, dst); - } else if (srcSize > BytesPerWord) { + } else if (srcSize > TargetBytesPerWord) { Site* low, *high, *other = pickSiteOrGrow(c, value, dst, &low, &high); other->freeze(c, value->nextWord); @@ -3065,7 +3067,7 @@ addReads(Context* c, Event* e, Value* v, unsigned size, { SingleRead* r = read(c, lowMask, lowSuccessor); addRead(c, e, v, r); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { r->high_ = v->nextWord; addRead(c, e, v->nextWord, highMask, highSuccessor); } @@ -3234,7 +3236,7 @@ class CallEvent: public Event { uint8_t typeMask; uint64_t planRegisterMask; c->arch->plan - ((flags & Compiler::Aligned) ? AlignedCall : Call, BytesPerWord, + ((flags & Compiler::Aligned) ? AlignedCall : Call, TargetBytesPerWord, &typeMask, &planRegisterMask, &thunk); assert(c, not thunk); @@ -3251,10 +3253,11 @@ class CallEvent: public Event { Value* v = stack->value; stack = stack->next; - if ((BytesPerWord == 8 and (v == 0 or (i >= 1 and stack->value == 0))) - or (BytesPerWord == 4 and v->nextWord != v)) + if ((TargetBytesPerWord == 8 + and (v == 0 or (i >= 1 and stack->value == 0))) + or (TargetBytesPerWord == 4 and v->nextWord != v)) { - assert(c, BytesPerWord == 8 or v->nextWord == stack->value); + assert(c, TargetBytesPerWord == 8 or v->nextWord == stack->value); RUNTIME_ARRAY_BODY(arguments)[i--] = stack->value; stack = stack->next; @@ -3405,7 +3408,7 @@ class CallEvent: public Event { op = Call; } - apply(c, op, BytesPerWord, address->source, address->source); + apply(c, op, TargetBytesPerWord, address->source, address->source); if (traceHandler) { traceHandler->handleTrace(codePromise(c, c->assembler->offset(true)), @@ -3436,7 +3439,7 @@ class CallEvent: public Event { if (resultSize and live(c, result)) { addSite(c, result, registerSite(c, c->arch->returnLow())); - if (resultSize > BytesPerWord and live(c, result->nextWord)) { + if (resultSize > TargetBytesPerWord and live(c, result->nextWord)) { addSite(c, result->nextWord, registerSite(c, c->arch->returnHigh())); } } @@ -3758,16 +3761,17 @@ class MoveEvent: public Event { bool noop = srcSelectSize >= dstSize; - if (dstSize > BytesPerWord) { + if (dstSize > TargetBytesPerWord) { grow(c, dst); } - if (srcSelectSize > BytesPerWord) { + if (srcSelectSize > TargetBytesPerWord) { maybeSplit(c, src); } addReads(c, this, src, srcSelectSize, srcLowMask, noop ? dst : 0, - srcHighMask, noop and dstSize > BytesPerWord ? dst->nextWord : 0); + srcHighMask, + noop and dstSize > TargetBytesPerWord ? dst->nextWord : 0); } virtual const char* name() { @@ -3791,43 +3795,46 @@ class MoveEvent: public Event { SiteMask dstLowMask(dstTypeMask, dstRegisterMask, AnyFrameIndex); SiteMask dstHighMask(dstTypeMask, dstRegisterMask >> 32, AnyFrameIndex); - if (srcSelectSize >= BytesPerWord - and dstSize >= BytesPerWord + if (srcSelectSize >= TargetBytesPerWord + and dstSize >= TargetBytesPerWord and srcSelectSize >= dstSize) { if (dst->target) { - if (dstSize > BytesPerWord - and src->source->registerSize(c) > BytesPerWord) + if (dstSize > TargetBytesPerWord + and src->source->registerSize(c) > TargetBytesPerWord) { apply(c, Move, srcSelectSize, src->source, src->source, dstSize, dst->target, dst->target); if (live(c, dst) == 0) { removeSite(c, dst, dst->target); - if (dstSize > BytesPerWord) { + if (dstSize > TargetBytesPerWord) { removeSite(c, dst->nextWord, dst->nextWord->target); } } } else { - maybeMove(c, Move, BytesPerWord, BytesPerWord, src, - BytesPerWord, dst, dstLowMask); - if (dstSize > BytesPerWord) { - maybeMove(c, Move, BytesPerWord, BytesPerWord, src->nextWord, - BytesPerWord, dst->nextWord, dstHighMask); + maybeMove(c, Move, TargetBytesPerWord, TargetBytesPerWord, src, + TargetBytesPerWord, dst, dstLowMask); + if (dstSize > TargetBytesPerWord) { + maybeMove + (c, Move, TargetBytesPerWord, TargetBytesPerWord, src->nextWord, + TargetBytesPerWord, dst->nextWord, dstHighMask); } } } else { Site* low = pickSiteOrMove(c, src, dst, 0, 0); - if (dstSize > BytesPerWord) { + if (dstSize > TargetBytesPerWord) { pickSiteOrMove(c, src->nextWord, dst->nextWord, low, 1); } } - } else if (srcSelectSize <= BytesPerWord and dstSize <= BytesPerWord) { + } else if (srcSelectSize <= TargetBytesPerWord + and dstSize <= TargetBytesPerWord) + { maybeMove(c, type, srcSize, srcSelectSize, src, dstSize, dst, dstLowMask); } else { - assert(c, srcSize == BytesPerWord); - assert(c, srcSelectSize == BytesPerWord); + assert(c, srcSize == TargetBytesPerWord); + assert(c, srcSelectSize == TargetBytesPerWord); if (dst->nextWord->target or live(c, dst->nextWord)) { assert(c, dstLowMask.typeMask & (1 << RegisterOperand)); @@ -3847,8 +3854,8 @@ class MoveEvent: public Event { srcb, dstb, src); } - apply(c, Move, BytesPerWord, src->source, src->source, - BytesPerWord, low, low); + apply(c, Move, TargetBytesPerWord, src->source, src->source, + TargetBytesPerWord, low, low); low->thaw(c, dst); @@ -3871,7 +3878,7 @@ class MoveEvent: public Event { srcb, dstb, dst, dst->nextWord); } - apply(c, Move, BytesPerWord, low, low, dstSize, low, high); + apply(c, Move, TargetBytesPerWord, low, low, dstSize, low, high); high->thaw(c, dst->nextWord); @@ -3972,7 +3979,7 @@ void freezeSource(Context* c, unsigned size, Value* v) { v->source->freeze(c, v); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { v->nextWord->source->freeze(c, v->nextWord); } } @@ -3981,7 +3988,7 @@ void thawSource(Context* c, unsigned size, Value* v) { v->source->thaw(c, v); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { v->nextWord->source->thaw(c, v->nextWord); } } @@ -4002,7 +4009,7 @@ class CombineEvent: public Event { { addReads(c, this, first, firstSize, firstLowMask, firstHighMask); - if (resultSize > BytesPerWord) { + if (resultSize > TargetBytesPerWord) { grow(c, result); } @@ -4227,7 +4234,7 @@ push(Context* c, unsigned footprint, Value* v) if (footprint > 1) { assert(c, footprint == 2); - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { maybeSplit(c, low); high = pushWord(c, low->nextWord); } else { @@ -4288,9 +4295,9 @@ pop(Context* c, unsigned footprint) high = low->next; } - assert(c, (BytesPerWord == 8 + assert(c, (TargetBytesPerWord == 8 and low->value->nextWord == low->value and high->value == 0) - or (BytesPerWord == 4 and low->value->nextWord == high->value)); + or (TargetBytesPerWord == 4 and low->value->nextWord == high->value)); #endif // not NDEBUG popWord(c); @@ -4331,7 +4338,7 @@ storeLocal(Context* c, unsigned footprint, Value* v, unsigned index, bool copy) highIndex = index; } - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { assert(c, v->nextWord != v); high = storeLocal(c, 1, v->nextWord, highIndex, false); @@ -4423,11 +4430,11 @@ appendCombine(Context* c, TernaryOperation type, intptr_t handler = c->client->getThunk (type, firstSize, resultSize, &threadParameter); - unsigned stackSize = ceiling(secondSize, BytesPerWord) - + ceiling(firstSize, BytesPerWord); + unsigned stackSize = ceiling(secondSize, TargetBytesPerWord) + + ceiling(firstSize, TargetBytesPerWord); - local::push(c, ceiling(secondSize, BytesPerWord), second); - local::push(c, ceiling(firstSize, BytesPerWord), first); + local::push(c, ceiling(secondSize, TargetBytesPerWord), second); + local::push(c, ceiling(firstSize, TargetBytesPerWord), first); if (threadParameter) { ++ stackSize; @@ -4467,7 +4474,7 @@ class TranslateEvent: public Event { { bool condensed = c->arch->alwaysCondensed(type); - if (resultSize > BytesPerWord) { + if (resultSize > TargetBytesPerWord) { grow(c, result); } @@ -4549,7 +4556,7 @@ appendTranslate(Context* c, BinaryOperation type, unsigned firstSize, if (thunk) { Stack* oldStack = c->stack; - local::push(c, ceiling(firstSize, BytesPerWord), first); + local::push(c, ceiling(firstSize, TargetBytesPerWord), first); Stack* argumentStack = c->stack; c->stack = oldStack; @@ -4559,7 +4566,7 @@ appendTranslate(Context* c, BinaryOperation type, unsigned firstSize, (c, ValueGeneral, constantSite (c, c->client->getThunk(type, firstSize, resultSize))), 0, 0, result, resultSize, argumentStack, - ceiling(firstSize, BytesPerWord), 0); + ceiling(firstSize, TargetBytesPerWord), 0); } else { append(c, new (c->zone->allocate(sizeof(TranslateEvent))) TranslateEvent @@ -4632,7 +4639,7 @@ class MemoryEvent: public Event { popRead(c, this, base); if (index) { - if (BytesPerWord == 8 and indexRegister != NoRegister) { + if (TargetBytesPerWord == 8 and indexRegister != NoRegister) { apply(c, Move, 4, index->source, index->source, 8, index->source, index->source); } @@ -4798,7 +4805,7 @@ class BranchEvent: public Event { uint8_t typeMask; uint64_t registerMask; - c->arch->planDestination(type, size, 0, 0, size, 0, 0, BytesPerWord, + c->arch->planDestination(type, size, 0, 0, size, 0, 0, TargetBytesPerWord, &typeMask, ®isterMask); addRead(c, this, address, SiteMask(typeMask, registerMask, AnyFrameIndex)); @@ -4821,7 +4828,7 @@ class BranchEvent: public Event { int64_t firstValue = firstConstant->value->value(); int64_t secondValue = secondConstant->value->value(); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { firstValue |= findConstantSite (c, first->nextWord)->value->value() << 32; secondValue |= findConstantSite @@ -4829,18 +4836,18 @@ class BranchEvent: public Event { } if (shouldJump(c, type, size, firstValue, secondValue)) { - apply(c, Jump, BytesPerWord, address->source, address->source); + apply(c, Jump, TargetBytesPerWord, address->source, address->source); } } else { freezeSource(c, size, first); freezeSource(c, size, second); - freezeSource(c, BytesPerWord, address); + freezeSource(c, TargetBytesPerWord, address); apply(c, type, size, first->source, first->nextWord->source, size, second->source, second->nextWord->source, - BytesPerWord, address->source, address->source); + TargetBytesPerWord, address->source, address->source); - thawSource(c, BytesPerWord, address); + thawSource(c, TargetBytesPerWord, address); thawSource(c, size, second); thawSource(c, size, first); } @@ -4872,7 +4879,7 @@ appendBranch(Context* c, TernaryOperation type, unsigned size, Value* first, c->arch->planSource(type, size, &firstTypeMask, &firstRegisterMask, size, &secondTypeMask, &secondRegisterMask, - BytesPerWord, &thunk); + TargetBytesPerWord, &thunk); if (thunk) { Stack* oldStack = c->stack; @@ -4883,8 +4890,8 @@ appendBranch(Context* c, TernaryOperation type, unsigned size, Value* first, assert(c, not threadParameter); - local::push(c, ceiling(size, BytesPerWord), second); - local::push(c, ceiling(size, BytesPerWord), first); + local::push(c, ceiling(size, TargetBytesPerWord), second); + local::push(c, ceiling(size, TargetBytesPerWord), first); Stack* argumentStack = c->stack; c->stack = oldStack; @@ -4893,7 +4900,7 @@ appendBranch(Context* c, TernaryOperation type, unsigned size, Value* first, appendCall (c, value (c, ValueGeneral, constantSite(c, handler)), 0, 0, result, 4, - argumentStack, ceiling(size, BytesPerWord) * 2, 0); + argumentStack, ceiling(size, TargetBytesPerWord) * 2, 0); appendBranch(c, thunkBranch(c, type), 4, value (c, ValueGeneral, constantSite(c, static_cast(0))), @@ -4920,7 +4927,7 @@ class JumpEvent: public Event { bool thunk; uint8_t typeMask; uint64_t registerMask; - c->arch->plan(type, BytesPerWord, &typeMask, ®isterMask, &thunk); + c->arch->plan(type, TargetBytesPerWord, &typeMask, ®isterMask, &thunk); assert(c, not thunk); @@ -4933,7 +4940,7 @@ class JumpEvent: public Event { virtual void compile(Context* c) { if (not unreachable(this)) { - apply(c, type, BytesPerWord, address->source, address->source); + apply(c, type, TargetBytesPerWord, address->source, address->source); } for (Read* r = reads; r; r = r->eventNext) { @@ -4992,7 +4999,7 @@ class BoundsCheckEvent: public Event { if (constant) { if (constant->value->value() < 0) { Assembler::Constant handlerConstant(resolved(c, handler)); - a->apply(Call, BytesPerWord, ConstantOperand, &handlerConstant); + a->apply(Call, TargetBytesPerWord, ConstantOperand, &handlerConstant); } } else { outOfBoundsPromise = codePromise(c, static_cast(0)); @@ -5000,7 +5007,7 @@ class BoundsCheckEvent: public Event { ConstantSite zero(resolved(c, 0)); ConstantSite oob(outOfBoundsPromise); apply(c, JumpIfLess, 4, &zero, &zero, 4, index->source, index->source, - BytesPerWord, &oob, &oob); + TargetBytesPerWord, &oob, &oob); } if (constant == 0 or constant->value->value() >= 0) { @@ -5011,20 +5018,20 @@ class BoundsCheckEvent: public Event { CodePromise* nextPromise = codePromise(c, static_cast(0)); - freezeSource(c, BytesPerWord, index); + freezeSource(c, TargetBytesPerWord, index); ConstantSite next(nextPromise); apply(c, JumpIfGreater, 4, index->source, index->source, 4, &length, - &length, BytesPerWord, &next, &next); + &length, TargetBytesPerWord, &next, &next); - thawSource(c, BytesPerWord, index); + thawSource(c, TargetBytesPerWord, index); if (constant == 0) { outOfBoundsPromise->offset = a->offset(); } Assembler::Constant handlerConstant(resolved(c, handler)); - a->apply(Call, BytesPerWord, ConstantOperand, &handlerConstant); + a->apply(Call, TargetBytesPerWord, ConstantOperand, &handlerConstant); nextPromise->offset = a->offset(); } @@ -6268,7 +6275,7 @@ class MyCompiler: public Compiler { virtual void save(unsigned footprint, Operand* value) { c.saved = cons(&c, static_cast(value), c.saved); - if (BytesPerWord == 4 and footprint > 1) { + if (TargetBytesPerWord == 4 and footprint > 1) { assert(&c, footprint == 2); assert(&c, static_cast(value)->nextWord); @@ -6329,9 +6336,10 @@ class MyCompiler: public Compiler { high = s->next; } - assert(&c, (BytesPerWord == 8 + assert(&c, (TargetBytesPerWord == 8 and low->value->nextWord == low->value and high->value == 0) - or (BytesPerWord == 4 and low->value->nextWord == high->value)); + or (TargetBytesPerWord == 4 + and low->value->nextWord == high->value)); #endif // not NDEBUG if (bigEndian) { @@ -6355,20 +6363,20 @@ class MyCompiler: public Compiler { bool bigEndian = c.arch->bigEndian(); unsigned footprint = 0; - unsigned size = BytesPerWord; + unsigned size = TargetBytesPerWord; RUNTIME_ARRAY(Value*, arguments, argumentCount); int index = 0; for (unsigned i = 0; i < argumentCount; ++i) { Value* o = va_arg(a, Value*); if (o) { - if (bigEndian and size > BytesPerWord) { + if (bigEndian and size > TargetBytesPerWord) { RUNTIME_ARRAY_BODY(arguments)[index++] = o->nextWord; } RUNTIME_ARRAY_BODY(arguments)[index] = o; - if ((not bigEndian) and size > BytesPerWord) { + if ((not bigEndian) and size > TargetBytesPerWord) { RUNTIME_ARRAY_BODY(arguments)[++index] = o->nextWord; } - size = BytesPerWord; + size = TargetBytesPerWord; ++ index; } else { size = 8; @@ -6427,7 +6435,7 @@ class MyCompiler: public Compiler { highIndex = index; } - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { initLocal(1, highIndex, type); Value* next = c.locals[highIndex].value; v->nextWord = next; @@ -6499,7 +6507,7 @@ class MyCompiler: public Compiler { virtual Operand* load(unsigned srcSize, unsigned srcSelectSize, Operand* src, unsigned dstSize) { - assert(&c, dstSize >= BytesPerWord); + assert(&c, dstSize >= TargetBytesPerWord); Value* dst = value(&c, static_cast(src)->type); appendMove(&c, Move, srcSize, srcSelectSize, static_cast(src), @@ -6510,7 +6518,7 @@ class MyCompiler: public Compiler { virtual Operand* loadz(unsigned srcSize, unsigned srcSelectSize, Operand* src, unsigned dstSize) { - assert(&c, dstSize >= BytesPerWord); + assert(&c, dstSize >= TargetBytesPerWord); Value* dst = value(&c, static_cast(src)->type); appendMove(&c, MoveZ, srcSize, srcSelectSize, static_cast(src), @@ -6786,7 +6794,7 @@ class MyCompiler: public Compiler { virtual Operand* shl(unsigned size, Operand* a, Operand* b) { assert(&c, static_cast(a)->type == ValueGeneral); Value* result = value(&c, ValueGeneral); - appendCombine(&c, ShiftLeft, BytesPerWord, static_cast(a), + appendCombine(&c, ShiftLeft, TargetBytesPerWord, static_cast(a), size, static_cast(b), size, result); return result; } @@ -6794,7 +6802,7 @@ class MyCompiler: public Compiler { virtual Operand* shr(unsigned size, Operand* a, Operand* b) { assert(&c, static_cast(a)->type == ValueGeneral); Value* result = value(&c, ValueGeneral); - appendCombine(&c, ShiftRight, BytesPerWord, static_cast(a), + appendCombine(&c, ShiftRight, TargetBytesPerWord, static_cast(a), size, static_cast(b), size, result); return result; } @@ -6803,8 +6811,8 @@ class MyCompiler: public Compiler { assert(&c, static_cast(a)->type == ValueGeneral); Value* result = value(&c, ValueGeneral); appendCombine - (&c, UnsignedShiftRight, BytesPerWord, static_cast(a), size, - static_cast(b), size, result); + (&c, UnsignedShiftRight, TargetBytesPerWord, static_cast(a), + size, static_cast(b), size, result); return result; } @@ -6933,7 +6941,7 @@ class MyCompiler: public Compiler { } virtual unsigned poolSize() { - return c.constantCount * BytesPerWord; + return c.constantCount * TargetBytesPerWord; } virtual void write() { @@ -6962,7 +6970,7 @@ class MyCompiler: public Compiler { new (n->promise->listen(sizeof(Listener))) Listener(target); } - i += BytesPerWord; + i += TargetBytesPerWord; } } diff --git a/src/heap.cpp b/src/heap.cpp index be26c9ea3e..8208cd7e73 100644 --- a/src/heap.cpp +++ b/src/heap.cpp @@ -19,12 +19,6 @@ namespace { namespace local { -// an object must survive TenureThreshold + 2 garbage collections -// before being copied to gen2 (must be at least 1): -const unsigned TenureThreshold = 3; - -const unsigned FixieTenureThreshold = TenureThreshold + 2; - const unsigned Top = ~static_cast(0); const unsigned InitialGen2CapacityInBytes = 4 * 1024 * 1024; @@ -546,11 +540,14 @@ class Fixie { return totalSize(size, hasMask); } + // be sure to update e.g. TargetFixieSizeInBytes in bootimage.cpp if + // you add/remove/change fields in this class: + uint8_t age; bool hasMask; bool marked; bool dirty; - unsigned size; + uint32_t size; Fixie* next; Fixie** handle; uintptr_t body_[0]; diff --git a/src/heap.h b/src/heap.h index ecb60d2637..838d88786c 100644 --- a/src/heap.h +++ b/src/heap.h @@ -16,6 +16,12 @@ namespace vm { +// an object must survive TenureThreshold + 2 garbage collections +// before being copied to gen2 (must be at least 1): +const unsigned TenureThreshold = 3; + +const unsigned FixieTenureThreshold = TenureThreshold + 2; + class Heap: public Allocator { public: enum CollectionType { diff --git a/src/interpret.cpp b/src/interpret.cpp index b5810a434e..255977c25d 100644 --- a/src/interpret.cpp +++ b/src/interpret.cpp @@ -665,7 +665,7 @@ store(Thread* t, unsigned index) BytesPerWord * 2); } -ExceptionHandler* +uint64_t findExceptionHandler(Thread* t, object method, unsigned ip) { PROTECT(t, method); @@ -674,7 +674,7 @@ findExceptionHandler(Thread* t, object method, unsigned ip) if (eht) { for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) { - ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i); + uint64_t eh = exceptionHandlerTableBody(t, eht, i); if (ip - 1 >= exceptionHandlerStart(eh) and ip - 1 < exceptionHandlerEnd(eh)) @@ -708,7 +708,7 @@ findExceptionHandler(Thread* t, object method, unsigned ip) return 0; } -ExceptionHandler* +uint64_t findExceptionHandler(Thread* t, int frame) { return findExceptionHandler(t, frameMethod(t, frame), frameIp(t, frame)); @@ -2673,7 +2673,7 @@ interpret3(Thread* t, const int base) pokeInt(t, t->frame + FrameIpOffset, t->ip); for (; frame >= base; popFrame(t)) { - ExceptionHandler* eh = findExceptionHandler(t, frame); + uint64_t eh = findExceptionHandler(t, frame); if (eh) { sp = frame + FrameFootprint; ip = exceptionHandlerIp(eh); diff --git a/src/machine.cpp b/src/machine.cpp index 9168c09019..0219894f77 100644 --- a/src/machine.cpp +++ b/src/machine.cpp @@ -1143,11 +1143,10 @@ parseFieldTable(Thread* t, Stream& s, object class_, object pool) addendum, class_); + unsigned size = fieldSize(t, code); if (flags & ACC_STATIC) { - unsigned size = fieldSize(t, code); - unsigned excess = (staticOffset % size) % BytesPerWord; - if (excess) { - staticOffset += BytesPerWord - excess; + while (staticOffset % size) { + ++ staticOffset; } fieldOffset(t, field) = staticOffset; @@ -1162,12 +1161,13 @@ parseFieldTable(Thread* t, Stream& s, object class_, object pool) classVmFlags(t, class_) |= HasFinalMemberFlag; } - while (memberOffset % fieldSize(t, code)) { + while (memberOffset % size) { ++ memberOffset; } fieldOffset(t, field) = memberOffset; - memberOffset += fieldSize(t, code); + + memberOffset += size; } set(t, fieldTable, ArrayBody + (i * BytesPerWord), field); @@ -1298,11 +1298,12 @@ parseCode(Thread* t, Stream& s, object pool) if (ehtLength) { object eht = makeExceptionHandlerTable(t, ehtLength); for (unsigned i = 0; i < ehtLength; ++i) { - ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i); - exceptionHandlerStart(eh) = s.read2(); - exceptionHandlerEnd(eh) = s.read2(); - exceptionHandlerIp(eh) = s.read2(); - exceptionHandlerCatchType(eh) = s.read2(); + unsigned start = s.read2(); + unsigned end = s.read2(); + unsigned ip = s.read2(); + unsigned catchType = s.read2(); + exceptionHandlerTableBody(t, eht, i) = exceptionHandler + (start, end, ip, catchType); } set(t, code, CodeExceptionHandlerTable, eht); @@ -1319,9 +1320,9 @@ parseCode(Thread* t, Stream& s, object pool) unsigned lntLength = s.read2(); object lnt = makeLineNumberTable(t, lntLength); for (unsigned i = 0; i < lntLength; ++i) { - LineNumber* ln = lineNumberTableBody(t, lnt, i); - lineNumberIp(ln) = s.read2(); - lineNumberLine(ln) = s.read2(); + unsigned ip = s.read2(); + unsigned line = s.read2(); + lineNumberTableBody(t, lnt, i) = lineNumber(ip, line); } set(t, code, CodeLineNumberTable, lnt); @@ -2217,9 +2218,9 @@ boot(Thread* t) #include "type-java-initializations.cpp" -#ifdef AVIAN_HEAPDUMP + //#ifdef AVIAN_HEAPDUMP # include "type-name-initializations.cpp" -#endif + //#endif } } @@ -3428,6 +3429,16 @@ parseClass(Thread* t, object loader, const uint8_t* data, unsigned size, updateClassTables(t, real, class_); + if (root(t, Machine::PoolMap)) { + object bootstrapClass = hashMapFind + (t, root(t, Machine::BootstrapClassMap), className(t, class_), + byteArrayHash, byteArrayEqual); + + hashMapInsert + (t, root(t, Machine::PoolMap), bootstrapClass ? bootstrapClass : real, + pool, objectHash); + } + return real; } @@ -4426,7 +4437,7 @@ vmAddressFromLine(Thread* t, object m, unsigned line) unsigned top = lineNumberTableLength(t, lnt); for(unsigned i = bottom; i < top; i++) { - LineNumber* ln = lineNumberTableBody(t, lnt, i); + uint64_t ln = lineNumberTableBody(t, lnt, i); if(lineNumberLine(ln) == line) return reinterpret_cast(lineNumberIp(ln)); else if(lineNumberLine(ln) > line) diff --git a/src/machine.h b/src/machine.h index 28a3a764f1..1b53c9f8d8 100644 --- a/src/machine.h +++ b/src/machine.h @@ -1253,6 +1253,7 @@ class Machine { MonitorMap, StringMap, ByteArrayMap, + PoolMap, ClassRuntimeDataTable, MethodRuntimeDataTable, JNIMethodTable, @@ -3283,14 +3284,20 @@ methodVirtual(Thread* t, object method) } inline unsigned -singletonMaskSize(unsigned count) +singletonMaskSize(unsigned count, unsigned bitsPerWord) { if (count) { - return ceiling(count + 2, BitsPerWord); + return ceiling(count + 2, bitsPerWord); } return 0; } +inline unsigned +singletonMaskSize(unsigned count) +{ + return singletonMaskSize(count, BitsPerWord); +} + inline unsigned singletonMaskSize(Thread* t, object singleton) { @@ -3315,11 +3322,17 @@ singletonMask(Thread* t, object singleton) (&singletonBody(t, singleton, singletonCount(t, singleton))); } +inline void +singletonMarkObject(uint32_t* mask, unsigned index) +{ + mask[(index + 2) / 32] + |= (static_cast(1) << ((index + 2) % 32)); +} + inline void singletonMarkObject(Thread* t, object singleton, unsigned index) { - singletonMask(t, singleton)[(index + 2) / 32] - |= (static_cast(1) << ((index + 2) % 32)); + singletonMarkObject(singletonMask(t, singleton), index); } inline bool @@ -3370,10 +3383,16 @@ singletonBit(Thread* t, object singleton, unsigned start, unsigned index) & (static_cast(1) << (index % BitsPerWord))) != 0; } +inline unsigned +poolMaskSize(unsigned count, unsigned bitsPerWord) +{ + return ceiling(count, bitsPerWord); +} + inline unsigned poolMaskSize(unsigned count) { - return ceiling(count, BitsPerWord); + return poolMaskSize(count, BitsPerWord); } inline unsigned @@ -3766,6 +3785,54 @@ methodClone(Thread* t, object method) methodCode(t, method)); } +inline uint64_t +exceptionHandler(uint64_t start, uint64_t end, uint64_t ip, uint64_t catchType) +{ + return (start << 48) | (end << 32) | (ip << 16) | catchType; +} + +inline unsigned +exceptionHandlerStart(uint64_t eh) +{ + return eh >> 48; +} + +inline unsigned +exceptionHandlerEnd(uint64_t eh) +{ + return (eh >> 32) & 0xFFFF; +} + +inline unsigned +exceptionHandlerIp(uint64_t eh) +{ + return (eh >> 16) & 0xFFFF; +} + +inline unsigned +exceptionHandlerCatchType(uint64_t eh) +{ + return eh & 0xFFFF; +} + +inline uint64_t +lineNumber(uint64_t ip, uint64_t line) +{ + return (ip << 32) | line; +} + +inline unsigned +lineNumberIp(uint64_t ln) +{ + return ln >> 32; +} + +inline unsigned +lineNumberLine(uint64_t ln) +{ + return ln & 0xFFFFFFFF; +} + inline FILE* errorLog(Thread* t) { diff --git a/src/process.cpp b/src/process.cpp index d2aebb131d..ac1c21917e 100644 --- a/src/process.cpp +++ b/src/process.cpp @@ -271,7 +271,7 @@ findLineNumber(Thread* t, object method, unsigned ip) unsigned top = lineNumberTableLength(t, lnt); for (unsigned span = top - bottom; span; span = top - bottom) { unsigned middle = bottom + (span / 2); - LineNumber* ln = lineNumberTableBody(t, lnt, middle); + uint64_t ln = lineNumberTableBody(t, lnt, middle); if (ip >= lineNumberIp(ln) and (middle + 1 == lineNumberTableLength(t, lnt) diff --git a/src/target.h b/src/target.h new file mode 100644 index 0000000000..91244b7ed6 --- /dev/null +++ b/src/target.h @@ -0,0 +1,66 @@ +/* Copyright (c) 2011, 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. */ + +#ifndef TARGET_H +#define TARGET_H + +#define TARGET_V1(v) (v) + +#ifdef TARGET_OPPOSITE_ENDIAN +# define TARGET_V2(v) \ + ((((v) >> 8) & 0xFF) | \ + (((v) << 8))) +# define TARGET_V4(v) \ + ((((v) >> 24) & 0x000000FF) | \ + (((v) >> 8) & 0x0000FF00) | \ + (((v) << 8) & 0x00FF0000) | \ + (((v) << 24))) +# define TARGET_V8(v) \ + (((static_cast(v) >> 56) & UINT64_C(0x00000000000000FF)) | \ + ((static_cast(v) >> 40) & UINT64_C(0x000000000000FF00)) | \ + ((static_cast(v) >> 24) & UINT64_C(0x0000000000FF0000)) | \ + ((static_cast(v) >> 8) & UINT64_C(0x00000000FF000000)) | \ + ((static_cast(v) << 8) & UINT64_C(0x000000FF00000000)) | \ + ((static_cast(v) << 24) & UINT64_C(0x0000FF0000000000)) | \ + ((static_cast(v) << 40) & UINT64_C(0x00FF000000000000)) | \ + ((static_cast(v) << 56))) +#else +# define TARGET_V2(v) (v) +# define TARGET_V4(v) (v) +# define TARGET_V8(v) (v) +#endif + +namespace vm { + +#ifdef TARGET_BYTES_PER_WORD +# if (TARGET_BYTES_PER_WORD == 8) +# define TARGET_VW(v) TARGET_V8(v) +typedef uint64_t target_uintptr_t; +typedef int64_t target_intptr_t; +const unsigned TargetBytesPerWord = 8; +# elif (TARGET_BYTES_PER_WORD == 4) +# define TARGET_VW(v) TARGET_V4(v) +typedef uint32_t target_uintptr_t; +typedef int32_t target_intptr_t; +const unsigned TargetBytesPerWord = 4; +# else +# error +# endif +#else +typedef uintptr_t target_uintptr_t; +typedef intptr_t target_intptr_t; +const unsigned TargetBytesPerWord = BytesPerWord; +#endif + +const unsigned TargetBitsPerWord = TargetBytesPerWord * 8; + +} // namespace vm + +#endif//TARGET_H diff --git a/src/type-generator.cpp b/src/type-generator.cpp index 626573af8e..eb0141f215 100644 --- a/src/type-generator.cpp +++ b/src/type-generator.cpp @@ -225,7 +225,6 @@ class Object { Scalar, Array, Method, - Pod, Type, Pair, Number, @@ -314,20 +313,18 @@ class List { class Scalar : public Object { public: Object* owner; - Object* typeObject; const char* typeName; const char* name; unsigned elementSize; bool noassert; bool nogc; - static Scalar* make(Object* owner, Object* typeObject, const char* typeName, - const char* name, unsigned size) + static Scalar* make(Object* owner, const char* typeName, const char* name, + unsigned size) { Scalar* o = allocate(); o->type = Object::Scalar; o->owner = owner; - o->typeObject = typeObject; o->typeName = typeName; o->name = name; o->elementSize = size; @@ -339,13 +336,12 @@ class Scalar : public Object { class Array : public Scalar { public: - static Array* make(Object* owner, Object* typeObject, const char* typeName, - const char* name, unsigned elementSize) + static Array* make(Object* owner, const char* typeName, const char* name, + unsigned elementSize) { Array* o = allocate(); o->type = Object::Array; o->owner = owner; - o->typeObject = typeObject; o->typeName = typeName; o->name = name; o->elementSize = elementSize; @@ -380,19 +376,6 @@ memberOwner(Object* o) } } -Object* -memberTypeObject(Object* o) -{ - switch (o->type) { - case Object::Scalar: - case Object::Array: - return static_cast(o)->typeObject; - - default: - UNREACHABLE; - } -} - const char* memberTypeName(Object* o) { @@ -406,6 +389,17 @@ memberTypeName(Object* o) } } +const char* +memberTypeEnumName(Object* o) +{ + const char* n = memberTypeName(o); + if (strcmp("void*", n) == 0) { + return "word"; + } else { + return n; + } +} + const char*& memberName(Object* o) { @@ -545,7 +539,7 @@ const char* typeName(Object* o) { switch (o->type) { - case Object::Type: case Object::Pod: + case Object::Type: return static_cast(o)->name; default: @@ -557,7 +551,7 @@ const char* typeJavaName(Object* o) { switch (o->type) { - case Object::Type: case Object::Pod: + case Object::Type: return static_cast(o)->javaName; default: @@ -569,7 +563,7 @@ Object* typeMembers(Object* o) { switch (o->type) { - case Object::Type: case Object::Pod: + case Object::Type: return static_cast(o)->members.first; default: @@ -605,10 +599,10 @@ void addMember(Object* o, Object* member) { switch (o->type) { - case Object::Type: case Object::Pod: + case Object::Type: if (member->type == Object::Array) { static_cast(o)->members.append - (Scalar::make(o, 0, "uintptr_t", "length", BytesPerWord)); + (Scalar::make(o, "uintptr_t", "length", BytesPerWord)); } static_cast(o)->members.append(member); break; @@ -822,7 +816,7 @@ declaration(const char* name, Object* declarations) for (Object* p = declarations; p; p = cdr(p)) { Object* o = car(p); switch (o->type) { - case Object::Type: case Object::Pod: + case Object::Type: if (equal(name, typeName(o))) return o; break; @@ -842,9 +836,6 @@ javaDeclaration(const char* name, Object* declarations) if (typeJavaName(o) and equal(name, typeJavaName(o))) return o; break; - case Object::Pod: - break; - default: UNREACHABLE; } } @@ -854,15 +845,11 @@ javaDeclaration(const char* name, Object* declarations) Object* derivationChain(Object* o) { - if (o->type == Object::Pod) { - return cons(o, 0); - } else { - Object* chain = 0; - for (Object* p = o; p; p = typeSuper(p)) { - chain = cons(p, chain); - } - return chain; + Object* chain = 0; + for (Object* p = o; p; p = typeSuper(p)) { + chain = cons(p, chain); } + return chain; } class MemberIterator { @@ -883,7 +870,7 @@ class MemberIterator { members(0), member(0), index_(-1), - offset_(type->type == Object::Pod ? 0 : BytesPerWord), + offset_(BytesPerWord), size_(0), padding_(0), alignment_(BytesPerWord) @@ -972,21 +959,6 @@ class MemberIterator { } }; -unsigned -typeSize(Object* o) -{ - switch (o->type) { - case Object::Pod: { - MemberIterator it(o); - while (it.hasMore()) it.next(); - return pad(it.offset() + it.space()); - } break; - - default: - UNREACHABLE; - } -} - bool namesPointer(const char* s) { @@ -996,7 +968,7 @@ namesPointer(const char* s) } unsigned -sizeOf(const char* type, Object* declarations) +sizeOf(const char* type) { if (equal(type, "object") or equal(type, "intptr_t") or equal(type, "uintptr_t")) @@ -1021,33 +993,29 @@ sizeOf(const char* type, Object* declarations) } else if (namesPointer(type)) { return BytesPerWord; } else { - Object* dec = declaration(type, declarations); - if (dec) return typeSize(dec); - fprintf(stderr, "unexpected type: %s\n", type); abort(); } } Object* -parseArray(Object* t, Object* p, Object* declarations) +parseArray(Object* t, Object* p) { const char* typeName = string(car(p)); p = cdr(p); const char* name = string(car(p)); - return Array::make(t, declaration(typeName, declarations), - typeName, name, sizeOf(typeName, declarations)); + return Array::make(t, typeName, name, sizeOf(typeName)); } Object* -parseMember(Object* t, Object* p, Object* declarations); +parseMember(Object* t, Object* p); Object* -parseMember(Object* t, Object* p, Object* declarations, bool* isNew) +parseMember(Object* t, Object* p, bool* isNew) { - Object* member = parseMember(t, p, declarations); + Object* member = parseMember(t, p); for (MemberIterator it(t); it.hasMore();) { Object* m = it.next(); if (equal(memberName(m), memberName(member))) { @@ -1063,34 +1031,32 @@ parseMember(Object* t, Object* p, Object* declarations, bool* isNew) } Object* -parseMember(Object* t, Object* p, Object* declarations) +parseMember(Object* t, Object* p) { const char* spec = string(car(p)); if (equal(spec, "array")) { - return parseArray(t, cdr(p), declarations); + return parseArray(t, cdr(p)); } else if (equal(spec, "noassert")) { bool isNew; - Object* member = parseMember(t, cdr(p), declarations, &isNew); + Object* member = parseMember(t, cdr(p), &isNew); memberNoAssert(member) = true; return isNew ? member : 0; } else if (equal(spec, "nogc")) { bool isNew; - Object* member = parseMember(t, cdr(p), declarations, &isNew); + Object* member = parseMember(t, cdr(p), &isNew); memberNoGC(member) = true; return isNew ? member : 0; } else if (equal(spec, "require")) { bool isNew; - Object* member = parseMember(t, cdr(p), declarations, &isNew); + Object* member = parseMember(t, cdr(p), &isNew); return isNew ? member : 0; } else if (equal(spec, "alias")) { bool isNew; - Object* member = parseMember(t, cdr(cdr(p)), declarations, &isNew); + Object* member = parseMember(t, cdr(cdr(p)), &isNew); memberName(member) = string(car(cdr(p))); return 0; } else { - return Scalar::make(t, declaration(spec, declarations), spec, - string(car(cdr(p))), - sizeOf(spec, declarations)); + return Scalar::make(t, spec, string(car(cdr(p))), sizeOf(spec)); } } @@ -1105,7 +1071,7 @@ parseSubdeclaration(Object* t, Object* p, Object* declarations) assert(typeSuper(t)); assert(typeSuper(t)->type == Object::Type); } else { - Object* member = parseMember(t, p, declarations); + Object* member = parseMember(t, p); if (member) { addMember(t, member); } @@ -1297,7 +1263,7 @@ parseJavaClass(Object* type, Stream* s, Object* declarations) const char* memberType = fieldType(spec); Object* member = Scalar::make - (type, 0, memberType, name, sizeOf(memberType, declarations)); + (type, memberType, name, sizeOf(memberType)); addMember(type, member); } @@ -1365,7 +1331,7 @@ parseType(Finder* finder, Object::ObjectType type, Object* p, if (type == Object::Type) { parseSubdeclaration(t, car(p), declarations); } else { - Object* member = parseMember(t, car(p), declarations); + Object* member = parseMember(t, car(p)); if (member) { assert(member->type == Object::Scalar); addMember(t, member); @@ -1390,8 +1356,6 @@ parseDeclaration(Finder* finder, Object* p, Object* declarations) const char* spec = string(car(p)); if (equal(spec, "type")) { return parseType(finder, Object::Type, cdr(p), declarations); - } else if (equal(spec, "pod")) { - return parseType(finder, Object::Pod, cdr(p), declarations); } else { fprintf(stderr, "unexpected declaration spec: %s\n", spec); abort(); @@ -1473,7 +1437,6 @@ void writeAccessor(Output* out, Object* member, Object* offset, bool unsafe = false) { const char* typeName = memberTypeName(member); - if (memberTypeObject(member)) typeName = capitalize(typeName); if (not unsafe) { out->write("const unsigned "); @@ -1491,22 +1454,12 @@ writeAccessor(Output* out, Object* member, Object* offset, bool unsafe = false) out->write("*"); } else { out->write(typeName); - if (member->type != Object::Scalar and memberTypeObject(member)) { - out->write("*"); - } else { - out->write("&"); - } + out->write("&"); } out->write("\n"); writeAccessorName(out, member, unsafe); - if (memberOwner(member)->type == Object::Pod) { - out->write("("); - out->write(capitalize(local::typeName(memberOwner(member)))); - out->write("*"); - } else { - out->write("(Thread* t UNUSED, object"); - } + out->write("(Thread* t UNUSED, object"); out->write(" o"); if (member->type != Object::Scalar) { out->write(", unsigned i"); @@ -1537,41 +1490,22 @@ writeAccessor(Output* out, Object* member, Object* offset, bool unsafe = false) out->write("*"); } else { out->write(typeName); - if (member->type != Object::Scalar and memberTypeObject(member)) { - out->write("*"); - } else { - out->write("&"); - } + out->write("&"); } out->write(">(reinterpret_cast(o)"); - if (endsWith("[0]", typeName) - or (member->type != Object::Scalar - and memberTypeObject(member))) - { - out->write(" + "); - } else { - out->write("["); - } + out->write("["); out->write(capitalize(local::typeName(memberOwner(member)))); out->write(capitalize(memberName(member))); if (member->type != Object::Scalar) { out->write(" + (i * "); - unsigned elementSize = (memberTypeObject(member) ? - typeSize(memberTypeObject(member)) : - sizeOf(memberTypeName(member), 0)); + unsigned elementSize = sizeOf(memberTypeName(member)); out->write(elementSize); out->write(")"); } - if (not endsWith("[0]", typeName) - and (member->type == Object::Scalar - or memberTypeObject(member) == 0)) - { - out->write("]"); - } - out->write(");\n}\n\n"); + out->write("]);\n}\n\n"); } Object* @@ -1615,39 +1549,13 @@ typeOffset(Object* type) return typeOffset(0, type); } -void -writePods(Output* out, Object* declarations) -{ - for (Object* p = declarations; p; p = cdr(p)) { - Object* o = car(p); - switch (o->type) { - case Object::Pod: { - out->write("const unsigned "); - out->write(capitalize(typeName(o))); - out->write("Size = "); - out->write(typeSize(o)); - out->write(";\n\n"); - - out->write("struct "); - out->write(capitalize(typeName(o))); - out->write(" { uint8_t body["); - out->write(capitalize(typeName(o))); - out->write("Size]; };\n\n"); - } break; - - default: break; - } - } -} - void writeAccessors(Output* out, Object* declarations) { for (Object* p = declarations; p; p = cdr(p)) { Object* o = car(p); switch (o->type) { - case Object::Type: - case Object::Pod: { + case Object::Type: { Object* offset = typeOffset (o, o->type == Object::Type ? typeSuper(o) : 0); for (MemberIterator it(o, true); it.hasMore();) { @@ -2031,15 +1939,6 @@ typeObjectMask(Object* type) case Object::Array: { if (memberGC(m)) { set(&mask, offset); - } else if (memberTypeObject(m) - and memberTypeObject(m)->type == Object::Pod) - { - for (MemberIterator it(memberTypeObject(m)); it.hasMore();) { - Object* m = it.next(); - if (memberGC(m)) { - set(&mask, offset + (it.offset() / BytesPerWord)); - } - } } } break; @@ -2207,13 +2106,73 @@ writeNameInitializations(Output* out, Object* declarations) } } +void +writeMap(Output* out, Object* type) +{ + for (MemberIterator it(type); it.hasMore();) { + Object* m = it.next(); + + switch (m->type) { + case Object::Scalar: { + out->write("Type_"); + out->write(memberTypeEnumName(m)); + } break; + + case Object::Array: { + out->write("Type_array, "); + out->write("Type_"); + out->write(memberTypeEnumName(m)); + } break; + + default: UNREACHABLE; + } + + out->write(", "); + } + + out->write("Type_none"); +} + +void +writeMaps(Output* out, Object* declarations) +{ + unsigned count = 0; + for (Object* p = declarations; p; p = cdr(p)) { + if (car(p)->type == Object::Type) { + ++ count; + } + } + + out->write("Type types[]["); + out->write(count); + out->write("] = {\n"); + bool wrote = false; + for (Object* p = declarations; p; p = cdr(p)) { + Object* o = car(p); + if (o->type == Object::Type) { + if (wrote) { + out->write(",\n"); + } else { + wrote = true; + } + + out->write("// "); + out->write(typeName(o)); + out->write("\n{ "); + writeMap(out, o); + out->write(" }"); + } + } + out->write("\n};"); +} + void usageAndExit(const char* command) { fprintf(stderr, "usage: %s " "{enums,declarations,constructors,initializations," - "java-initializations,name-initializations}\n", + "java-initializations,name-initializations,maps}\n", command); exit(-1); } @@ -2243,7 +2202,8 @@ main(int ac, char** av) or local::equal(av[4], "constructors") or local::equal(av[4], "initializations") or local::equal(av[4], "java-initializations") - or local::equal(av[4], "name-initializations"))) + or local::equal(av[4], "name-initializations") + or local::equal(av[4], "maps"))) { local::usageAndExit(av[0]); } @@ -2301,7 +2261,6 @@ main(int ac, char** av) out.Output::write(local::typeCount(declarations)); out.write(";\n\n"); - local::writePods(&out, declarations); local::writeAccessors(&out, declarations); local::writeSizes(&out, declarations); local::writeInitializerDeclarations(&out, declarations); @@ -2315,6 +2274,8 @@ main(int ac, char** av) local::writeJavaInitializations(&out, declarations); } else if (local::equal(av[4], "name-initializations")) { local::writeNameInitializations(&out, declarations); + } else if (local::equal(av[4], "maps")) { + local::writeMaps(&out, declarations); } return 0; diff --git a/src/types.def b/src/types.def index 2178c3a7fa..c95ed03a0a 100644 --- a/src/types.def +++ b/src/types.def @@ -64,23 +64,13 @@ (type region (void* region) - (unsigned position)) - -(pod exceptionHandler - (uint16_t start) - (uint16_t end) - (uint16_t ip) - (uint16_t catchType)) + (uint32_t position)) (type exceptionHandlerTable - (array exceptionHandler body)) - -(pod lineNumber - (uint32_t ip) - (uint32_t line)) + (array uint64_t body)) (type lineNumberTable - (array lineNumber body)) + (array uint64_t body)) (type code (object pool) @@ -126,7 +116,7 @@ (type traceElement (object method) - (int ip)) + (int32_t ip)) (type treeNode (object value) @@ -155,7 +145,7 @@ (void* waitTail) (object acquireHead) (object acquireTail) - (unsigned depth)) + (uint32_t depth)) (type monitorNode (void* value) diff --git a/src/x86.cpp b/src/x86.cpp index 670c5482d3..5a90ede640 100644 --- a/src/x86.cpp +++ b/src/x86.cpp @@ -9,6 +9,7 @@ details. */ #include "assembler.h" +#include "target.h" #include "vector.h" #define CAST1(x) reinterpret_cast(x) @@ -60,17 +61,17 @@ enum { }; const unsigned GeneralRegisterMask -= BytesPerWord == 4 ? 0x000000ff : 0x0000ffff; += TargetBytesPerWord == 4 ? 0x000000ff : 0x0000ffff; const unsigned FloatRegisterMask -= BytesPerWord == 4 ? 0x00ff0000 : 0xffff0000; += TargetBytesPerWord == 4 ? 0x00ff0000 : 0xffff0000; const unsigned FrameHeaderSize = (UseFramePointer ? 2 : 1); const int LongJumpRegister = r10; const unsigned StackAlignmentInBytes = 16; -const unsigned StackAlignmentInWords = StackAlignmentInBytes / BytesPerWord; +const unsigned StackAlignmentInWords = StackAlignmentInBytes / TargetBytesPerWord; bool isInt8(intptr_t v) @@ -478,7 +479,7 @@ detectFeature(unsigned ecx, unsigned edx); bool useSSE(ArchitectureContext* c) { - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { // amd64 implies SSE2 support return true; } else if (c->useNativeFeatures) { @@ -502,7 +503,7 @@ useSSE(ArchitectureContext* c) void maybeRex(Context* c, unsigned size, int a, int index, int base, bool always) { - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { uint8_t byte; if (size == 8) { byte = REX_W; @@ -656,7 +657,7 @@ storeLoadBarrier(Context* c) } else { // lock addq $0x0,(%rsp): c->code.append(0xf0); - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { c->code.append(0x48); } c->code.append(0x83); @@ -746,7 +747,7 @@ callR(Context*, unsigned, Assembler::Register*); void callC(Context* c, unsigned size UNUSED, Assembler::Constant* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); unconditional(c, 0xe8, a); } @@ -754,9 +755,9 @@ callC(Context* c, unsigned size UNUSED, Assembler::Constant* a) void longCallC(Context* c, unsigned size, Assembler::Constant* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { Assembler::Register r(LongJumpRegister); moveCR2(c, size, a, size, &r, 11); callR(c, size, &r); @@ -768,7 +769,7 @@ longCallC(Context* c, unsigned size, Assembler::Constant* a) void jumpR(Context* c, unsigned size UNUSED, Assembler::Register* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); maybeRex(c, 4, a); opcode(c, 0xff, 0xe0 + regCode(a)); @@ -777,7 +778,7 @@ jumpR(Context* c, unsigned size UNUSED, Assembler::Register* a) void jumpC(Context* c, unsigned size UNUSED, Assembler::Constant* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); unconditional(c, 0xe9, a); } @@ -785,7 +786,7 @@ jumpC(Context* c, unsigned size UNUSED, Assembler::Constant* a) void jumpM(Context* c, unsigned size UNUSED, Assembler::Memory* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); maybeRex(c, 4, a); opcode(c, 0xff); @@ -795,9 +796,9 @@ jumpM(Context* c, unsigned size UNUSED, Assembler::Memory* a) void longJumpC(Context* c, unsigned size, Assembler::Constant* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { Assembler::Register r(LongJumpRegister); moveCR2(c, size, a, size, &r, 11); jumpR(c, size, &r); @@ -809,7 +810,7 @@ longJumpC(Context* c, unsigned size, Assembler::Constant* a) void callR(Context* c, unsigned size UNUSED, Assembler::Register* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); // maybeRex.W has no meaning here so we disable it maybeRex(c, 4, a); @@ -819,7 +820,7 @@ callR(Context* c, unsigned size UNUSED, Assembler::Register* a) void callM(Context* c, unsigned size UNUSED, Assembler::Memory* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); maybeRex(c, 4, a); opcode(c, 0xff); @@ -836,9 +837,9 @@ alignedCallC(Context* c, unsigned size, Assembler::Constant* a) void alignedLongCallC(Context* c, unsigned size, Assembler::Constant* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { new (c->zone->allocate(sizeof(AlignmentPadding))) AlignmentPadding(c, 2, 8); longCallC(c, size, a); @@ -857,9 +858,9 @@ alignedJumpC(Context* c, unsigned size, Assembler::Constant* a) void alignedLongJumpC(Context* c, unsigned size, Assembler::Constant* a) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { new (c->zone->allocate(sizeof(AlignmentPadding))) AlignmentPadding(c, 2, 8); longJumpC(c, size, a); @@ -871,7 +872,7 @@ alignedLongJumpC(Context* c, unsigned size, Assembler::Constant* a) void pushR(Context* c, unsigned size, Assembler::Register* a) { - if (BytesPerWord == 4 and size == 8) { + if (TargetBytesPerWord == 4 and size == 8) { Assembler::Register ah(a->high); pushR(c, 4, &ah); @@ -889,7 +890,7 @@ moveRR(Context* c, unsigned aSize, Assembler::Register* a, void popR(Context* c, unsigned size, Assembler::Register* a) { - if (BytesPerWord == 4 and size == 8) { + if (TargetBytesPerWord == 4 and size == 8) { Assembler::Register ah(a->high); popR(c, 4, a); @@ -897,7 +898,7 @@ popR(Context* c, unsigned size, Assembler::Register* a) } else { maybeRex(c, 4, a); opcode(c, 0x58 + regCode(a)); - if (BytesPerWord == 8 and size == 4) { + if (TargetBytesPerWord == 8 and size == 4) { moveRR(c, 4, a, 8, a); } } @@ -910,7 +911,7 @@ addCarryCR(Context* c, unsigned size, Assembler::Constant* a, void negateR(Context* c, unsigned size, Assembler::Register* a) { - if (BytesPerWord == 4 and size == 8) { + if (TargetBytesPerWord == 4 and size == 8) { assert(c, a->low == rax and a->high == rdx); ResolvedPromise zeroPromise(0); @@ -940,7 +941,7 @@ void moveCR2(Context* c, UNUSED unsigned aSize, Assembler::Constant* a, UNUSED unsigned bSize, Assembler::Register* b, unsigned promiseOffset) { - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { int64_t v = a->value->value(); ResolvedPromise high((v >> 32) & 0xFFFFFFFF); @@ -954,13 +955,13 @@ moveCR2(Context* c, UNUSED unsigned aSize, Assembler::Constant* a, moveCR(c, 4, &al, 4, b); moveCR(c, 4, &ah, 4, &bh); } else { - maybeRex(c, BytesPerWord, b); + maybeRex(c, TargetBytesPerWord, b); opcode(c, 0xb8 + regCode(b)); if (a->value->resolved()) { c->code.appendAddress(a->value->value()); } else { appendImmediateTask - (c, a->value, offset(c), BytesPerWord, promiseOffset); + (c, a->value, offset(c), TargetBytesPerWord, promiseOffset); c->code.appendAddress(static_cast(0)); } } @@ -1008,7 +1009,7 @@ void sseMoveCR(Context* c, unsigned aSize, Assembler::Constant* a, unsigned bSize, Assembler::Register* b) { - assert(c, aSize <= BytesPerWord); + assert(c, aSize <= TargetBytesPerWord); Assembler::Register tmp(c->client->acquireTemporary(GeneralRegisterMask)); moveCR2(c, aSize, a, aSize, &tmp, 0); sseMoveRR(c, aSize, &tmp, bSize, b); @@ -1031,7 +1032,7 @@ swapRR(Context* c, unsigned aSize UNUSED, Assembler::Register* a, unsigned bSize UNUSED, Assembler::Register* b) { assert(c, aSize == bSize); - assert(c, aSize == BytesPerWord); + assert(c, aSize == TargetBytesPerWord); alwaysRex(c, aSize, a, b); opcode(c, 0x87); @@ -1047,7 +1048,7 @@ moveRR(Context* c, unsigned aSize, Assembler::Register* a, return; } - if (BytesPerWord == 4 and aSize == 8 and bSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8 and bSize == 8) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1065,11 +1066,11 @@ moveRR(Context* c, unsigned aSize, Assembler::Register* a, } else { switch (aSize) { case 1: - if (BytesPerWord == 4 and a->low > rbx) { + if (TargetBytesPerWord == 4 and a->low > rbx) { assert(c, b->low <= rbx); - moveRR(c, BytesPerWord, a, BytesPerWord, b); - moveRR(c, 1, b, BytesPerWord, b); + moveRR(c, TargetBytesPerWord, a, TargetBytesPerWord, b); + moveRR(c, 1, b, TargetBytesPerWord, b); } else { alwaysRex(c, aSize, b, a); opcode(c, 0x0f, 0xbe); @@ -1085,7 +1086,7 @@ moveRR(Context* c, unsigned aSize, Assembler::Register* a, case 4: if (bSize == 8) { - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { alwaysRex(c, bSize, b, a); opcode(c, 0x63); modrm(c, 0xc0, a, b); @@ -1125,7 +1126,7 @@ sseMoveMR(Context* c, unsigned aSize, Assembler::Memory* a, { assert(c, aSize >= 4); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { opcode(c, 0xf3); opcode(c, 0x0f, 0x7e); modrmSibImm(c, b, a); @@ -1160,7 +1161,7 @@ moveMR(Context* c, unsigned aSize, Assembler::Memory* a, break; case 4: - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { maybeRex(c, bSize, b, a); opcode(c, 0x63); modrmSibImm(c, b, a); @@ -1179,7 +1180,7 @@ moveMR(Context* c, unsigned aSize, Assembler::Memory* a, break; case 8: - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { Assembler::Memory ah(a->base, a->offset + 4, a->index, a->scale); Assembler::Register bh(b->high); @@ -1203,7 +1204,7 @@ sseMoveRM(Context* c, unsigned aSize, Assembler::Register* a, assert(c, aSize >= 4); assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { opcode(c, 0x66); opcode(c, 0x0f, 0xd6); modrmSibImm(c, a, b); @@ -1241,7 +1242,7 @@ moveRM(Context* c, unsigned aSize, Assembler::Register* a, break; case 4: - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { maybeRex(c, bSize, a, b); opcode(c, 0x89); modrmSibImm(c, a, b); @@ -1253,7 +1254,7 @@ moveRM(Context* c, unsigned aSize, Assembler::Register* a, break; case 8: - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { maybeRex(c, bSize, a, b); opcode(c, 0x89); modrmSibImm(c, a, b); @@ -1274,7 +1275,7 @@ void moveAR(Context* c, unsigned aSize, Assembler::Address* a, unsigned bSize, Assembler::Register* b) { - assert(c, BytesPerWord == 8 or (aSize == 4 and bSize == 4)); + assert(c, TargetBytesPerWord == 8 or (aSize == 4 and bSize == 4)); Assembler::Constant constant(a->address); Assembler::Memory memory(b->low, 0, -1, 0); @@ -1323,7 +1324,7 @@ moveCM(Context* c, unsigned aSize UNUSED, Assembler::Constant* a, break; case 8: { - if (BytesPerWord == 8) { + if (TargetBytesPerWord == 8) { if (a->value->resolved() and isInt32(a->value->value())) { maybeRex(c, bSize, b); opcode(c, 0xc7); @@ -1370,7 +1371,7 @@ void moveZMR(Context* c, unsigned aSize UNUSED, Assembler::Memory* a, unsigned bSize UNUSED, Assembler::Register* b) { - assert(c, bSize == BytesPerWord); + assert(c, bSize == TargetBytesPerWord); assert(c, aSize == 2); maybeRex(c, bSize, b, a); @@ -1382,7 +1383,7 @@ void addCarryRR(Context* c, unsigned size, Assembler::Register* a, Assembler::Register* b) { - assert(c, BytesPerWord == 8 or size == 4); + assert(c, TargetBytesPerWord == 8 or size == 4); maybeRex(c, size, a, b); opcode(c, 0x11); @@ -1395,7 +1396,7 @@ addRR(Context* c, unsigned aSize, Assembler::Register* a, { assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1432,7 +1433,7 @@ addCR(Context* c, unsigned aSize, Assembler::Constant* a, int64_t v = a->value->value(); if (v) { - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { ResolvedPromise high((v >> 32) & 0xFFFFFFFF); Assembler::Constant ah(&high); @@ -1468,7 +1469,7 @@ void subtractBorrowCR(Context* c, unsigned size UNUSED, Assembler::Constant* a, Assembler::Register* b) { - assert(c, BytesPerWord == 8 or size == 4); + assert(c, TargetBytesPerWord == 8 or size == 4); int64_t v = a->value->value(); if (isInt8(v)) { @@ -1491,7 +1492,7 @@ subtractCR(Context* c, unsigned aSize, Assembler::Constant* a, int64_t v = a->value->value(); if (v) { - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { ResolvedPromise high((v >> 32) & 0xFFFFFFFF); Assembler::Constant ah(&high); @@ -1527,7 +1528,7 @@ void subtractBorrowRR(Context* c, unsigned size, Assembler::Register* a, Assembler::Register* b) { - assert(c, BytesPerWord == 8 or size == 4); + assert(c, TargetBytesPerWord == 8 or size == 4); maybeRex(c, size, a, b); opcode(c, 0x19); @@ -1540,7 +1541,7 @@ subtractRR(Context* c, unsigned aSize, Assembler::Register* a, { assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1560,7 +1561,7 @@ andRR(Context* c, unsigned aSize, Assembler::Register* a, assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1581,7 +1582,7 @@ andCR(Context* c, unsigned aSize, Assembler::Constant* a, int64_t v = a->value->value(); - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { ResolvedPromise high((v >> 32) & 0xFFFFFFFF); Assembler::Constant ah(&high); @@ -1618,7 +1619,7 @@ orRR(Context* c, unsigned aSize, Assembler::Register* a, { assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1639,7 +1640,7 @@ orCR(Context* c, unsigned aSize, Assembler::Constant* a, int64_t v = a->value->value(); if (v) { - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { ResolvedPromise high((v >> 32) & 0xFFFFFFFF); Assembler::Constant ah(&high); @@ -1675,7 +1676,7 @@ void xorRR(Context* c, unsigned aSize, Assembler::Register* a, unsigned bSize UNUSED, Assembler::Register* b) { - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1696,7 +1697,7 @@ xorCR(Context* c, unsigned aSize, Assembler::Constant* a, int64_t v = a->value->value(); if (v) { - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { ResolvedPromise high((v >> 32) & 0xFFFFFFFF); Assembler::Constant ah(&high); @@ -1735,7 +1736,7 @@ multiplyRR(Context* c, unsigned aSize, Assembler::Register* a, assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { assert(c, b->high == rdx); assert(c, b->low != rax); assert(c, a->low != rax); @@ -1870,7 +1871,7 @@ compareRR(Context* c, unsigned aSize, Assembler::Register* a, unsigned bSize UNUSED, Assembler::Register* b) { assert(c, aSize == bSize); - assert(c, aSize <= BytesPerWord); + assert(c, aSize <= TargetBytesPerWord); maybeRex(c, aSize, a, b); opcode(c, 0x39); @@ -1882,7 +1883,7 @@ compareCR(Context* c, unsigned aSize, Assembler::Constant* a, unsigned bSize, Assembler::Register* b) { assert(c, aSize == bSize); - assert(c, BytesPerWord == 8 or aSize == 4); + assert(c, TargetBytesPerWord == 8 or aSize == 4); if (a->value->resolved() and isInt32(a->value->value())) { int64_t v = a->value->value(); @@ -1907,9 +1908,9 @@ compareRM(Context* c, unsigned aSize, Assembler::Register* a, unsigned bSize UNUSED, Assembler::Memory* b) { assert(c, aSize == bSize); - assert(c, BytesPerWord == 8 or aSize == 4); + assert(c, TargetBytesPerWord == 8 or aSize == 4); - if (BytesPerWord == 8 and aSize == 4) { + if (TargetBytesPerWord == 8 and aSize == 4) { moveRR(c, 4, a, 8, a); } maybeRex(c, bSize, a, b); @@ -1922,7 +1923,7 @@ compareCM(Context* c, unsigned aSize, Assembler::Constant* a, unsigned bSize, Assembler::Memory* b) { assert(c, aSize == bSize); - assert(c, BytesPerWord == 8 or aSize == 4); + assert(c, TargetBytesPerWord == 8 or aSize == 4); if (a->value->resolved()) { int64_t v = a->value->value(); @@ -2048,7 +2049,7 @@ branchRR(Context* c, TernaryOperation op, unsigned size, if (isFloatBranch(op)) { compareFloatRR(c, size, a, size, b); branchFloat(c, op, target); - } else if (size > BytesPerWord) { + } else if (size > TargetBytesPerWord) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -2066,7 +2067,7 @@ branchCR(Context* c, TernaryOperation op, unsigned size, { assert(c, not isFloatBranch(op)); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { int64_t v = a->value->value(); ResolvedPromise low(v & ~static_cast(0)); @@ -2090,7 +2091,7 @@ branchRM(Context* c, TernaryOperation op, unsigned size, Assembler::Constant* target) { assert(c, not isFloatBranch(op)); - assert(c, size <= BytesPerWord); + assert(c, size <= TargetBytesPerWord); compareRM(c, size, a, size, b); branch(c, op, target); @@ -2102,7 +2103,7 @@ branchCM(Context* c, TernaryOperation op, unsigned size, Assembler::Constant* target) { assert(c, not isFloatBranch(op)); - assert(c, size <= BytesPerWord); + assert(c, size <= TargetBytesPerWord); compareCM(c, size, a, size, b); branch(c, op, target); @@ -2114,7 +2115,7 @@ multiplyCR(Context* c, unsigned aSize, Assembler::Constant* a, { assert(c, aSize == bSize); - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { const uint32_t mask = GeneralRegisterMask & ~((1 << rax) | (1 << rdx)); Assembler::Register tmp(c->client->acquireTemporary(mask), c->client->acquireTemporary(mask)); @@ -2182,7 +2183,7 @@ remainderRR(Context* c, unsigned aSize, Assembler::Register* a, opcode(c, 0xf7, 0xf8 + regCode(a)); Assembler::Register dx(rdx); - moveRR(c, BytesPerWord, &dx, BytesPerWord, b); + moveRR(c, TargetBytesPerWord, &dx, TargetBytesPerWord, b); } void @@ -2194,7 +2195,7 @@ doShift(Context* c, UNUSED void (*shift) { int64_t v = a->value->value(); - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { c->client->save(rcx); Assembler::Register cx(rcx); @@ -2219,7 +2220,7 @@ shiftLeftRR(Context* c, UNUSED unsigned aSize, Assembler::Register* a, { assert(c, a->low == rcx); - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { // shld opcode(c, 0x0f, 0xa5); modrm(c, 0xc0, b->high, b->low); @@ -2255,7 +2256,7 @@ shiftRightRR(Context* c, UNUSED unsigned aSize, Assembler::Register* a, unsigned bSize, Assembler::Register* b) { assert(c, a->low == rcx); - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { // shrd opcode(c, 0x0f, 0xad); modrm(c, 0xc0, b->low, b->high); @@ -2295,7 +2296,7 @@ unsignedShiftRightRR(Context* c, UNUSED unsigned aSize, Assembler::Register* a, { assert(c, a->low == rcx); - if (BytesPerWord == 4 and bSize == 8) { + if (TargetBytesPerWord == 4 and bSize == 8) { // shrd opcode(c, 0x0f, 0xad); modrm(c, 0xc0, b->low, b->high); @@ -2556,7 +2557,7 @@ nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED, uint8_t* instruction = static_cast(*ip); // skip stack overflow check, if present: - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { if (*start == 0x39) { start += 11; } @@ -2571,7 +2572,7 @@ nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED, if (UseFramePointer) { // skip preamble - start += (BytesPerWord == 4 ? 3 : 4); + start += (TargetBytesPerWord == 4 ? 3 : 4); if (instruction <= start or *instruction == 0x5d) { *ip = static_cast(*stack)[1]; @@ -2596,13 +2597,13 @@ nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED, // check for post-non-tail-call stack adjustment of the form "add // $offset,%rsp": - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { if ((*instruction == 0x83 or *instruction == 0x81) and instruction[1] == 0xec) { offset -= (*instruction == 0x83 ? instruction[2] : read4(instruction + 2)) - / BytesPerWord; + / TargetBytesPerWord; } } else if (*instruction == 0x48 and (instruction[1] == 0x83 or instruction[1] == 0x81) @@ -2610,7 +2611,7 @@ nextFrame(ArchitectureContext* c UNUSED, uint8_t* start, unsigned size UNUSED, { offset -= (instruction[1] == 0x83 ? instruction[3] : read4(instruction + 3)) - / BytesPerWord; + / TargetBytesPerWord; } // todo: check for and handle tail calls @@ -2773,7 +2774,7 @@ class MyArchitecture: public Assembler::Architecture { } virtual int returnHigh() { - return (BytesPerWord == 4 ? rdx : NoRegister); + return (TargetBytesPerWord == 4 ? rdx : NoRegister); } virtual int virtualCallTarget() { @@ -2807,7 +2808,7 @@ class MyArchitecture: public Assembler::Architecture { } virtual unsigned frameFootprint(unsigned footprint) { -#ifdef PLATFORM_WINDOWS +#ifdef TARGET_PLATFORM_WINDOWS return max(footprint, StackAlignmentInWords); #else return max(footprint > argumentRegisterCount() ? @@ -2829,18 +2830,18 @@ class MyArchitecture: public Assembler::Architecture { } virtual unsigned argumentRegisterCount() { -#ifdef PLATFORM_WINDOWS - if (BytesPerWord == 8) return 4; else +#ifdef TARGET_PLATFORM_WINDOWS + if (TargetBytesPerWord == 8) return 4; else #else - if (BytesPerWord == 8) return 6; else + if (TargetBytesPerWord == 8) return 6; else #endif return 0; } virtual int argumentRegister(unsigned index) { - assert(&c, BytesPerWord == 8); + assert(&c, TargetBytesPerWord == 8); switch (index) { -#ifdef PLATFORM_WINDOWS +#ifdef TARGET_PLATFORM_WINDOWS case 0: return rcx; case 1: @@ -2913,7 +2914,7 @@ class MyArchitecture: public Assembler::Architecture { assertAlignment = false; } - if (BytesPerWord == 4 or op == Call or op == Jump) { + if (TargetBytesPerWord == 4 or op == Call or op == Jump) { uint8_t* instruction = static_cast(returnAddress) - 5; assert(&c, ((op == Call or op == LongCall) and *instruction == 0xE8) @@ -2945,8 +2946,9 @@ class MyArchitecture: public Assembler::Architecture { } } - virtual void setConstant(void* dst, uintptr_t constant) { - memcpy(dst, &constant, BytesPerWord); + virtual void setConstant(void* dst, uint64_t constant) { + target_uintptr_t v = TARGET_VW(constant); + memcpy(dst, &v, TargetBytesPerWord); } virtual unsigned alignFrameSize(unsigned sizeInWords) { @@ -3040,7 +3042,7 @@ class MyArchitecture: public Assembler::Architecture { break; case Absolute: - if (aSize <= BytesPerWord) { + if (aSize <= TargetBytesPerWord) { *aTypeMask = (1 << RegisterOperand); *aRegisterMask = (static_cast(1) << rax); } else { @@ -3089,7 +3091,7 @@ class MyArchitecture: public Assembler::Architecture { break; case Float2Int: - if (useSSE(&c) and bSize <= BytesPerWord) { + if (useSSE(&c) and bSize <= TargetBytesPerWord) { *aTypeMask = (1 << RegisterOperand) | (1 << MemoryOperand); *aRegisterMask = (static_cast(FloatRegisterMask) << 32) | FloatRegisterMask; @@ -3099,7 +3101,7 @@ class MyArchitecture: public Assembler::Architecture { break; case Int2Float: - if (useSSE(&c) and aSize <= BytesPerWord) { + if (useSSE(&c) and aSize <= TargetBytesPerWord) { *aTypeMask = (1 << RegisterOperand) | (1 << MemoryOperand); *aRegisterMask = GeneralRegisterMask | (static_cast(GeneralRegisterMask) << 32); @@ -3112,7 +3114,7 @@ class MyArchitecture: public Assembler::Architecture { *aTypeMask = ~0; *aRegisterMask = ~static_cast(0); - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { if (aSize == 4 and bSize == 8) { *aTypeMask = (1 << RegisterOperand) | (1 << MemoryOperand); const uint32_t mask @@ -3188,7 +3190,7 @@ class MyArchitecture: public Assembler::Architecture { *bTypeMask = (1 << RegisterOperand) | (1 << MemoryOperand); } - if (BytesPerWord == 4) { + if (TargetBytesPerWord == 4) { if (aSize == 4 and bSize == 8) { *bRegisterMask = (static_cast(1) << (rdx + 32)) | (static_cast(1) << rax); @@ -3223,7 +3225,7 @@ class MyArchitecture: public Assembler::Architecture { *tmpRegisterMask = GeneralRegisterMask | (static_cast(GeneralRegisterMask) << 32); } else if (dstTypeMask & (1 << RegisterOperand)) { - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { // can't move directly from FPR to GPR or vice-versa for // values larger than the GPR size if (dstRegisterMask & FloatRegisterMask) { @@ -3239,7 +3241,7 @@ class MyArchitecture: public Assembler::Architecture { if (dstRegisterMask & FloatRegisterMask) { // can't move directly from constant to FPR *srcTypeMask &= ~(1 << ConstantOperand); - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { *tmpTypeMask = 1 << MemoryOperand; } else { *tmpTypeMask = (1 << RegisterOperand) | (1 << MemoryOperand); @@ -3290,7 +3292,7 @@ class MyArchitecture: public Assembler::Architecture { break; case Multiply: - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { const uint32_t mask = GeneralRegisterMask & ~((1 << rax) | (1 << rdx)); *aRegisterMask = (static_cast(mask) << 32) | mask; *bRegisterMask = (static_cast(1) << (rdx + 32)) | mask; @@ -3301,7 +3303,7 @@ class MyArchitecture: public Assembler::Architecture { break; case Divide: - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { *thunk = true; } else { *aTypeMask = (1 << RegisterOperand); @@ -3311,7 +3313,7 @@ class MyArchitecture: public Assembler::Architecture { break; case Remainder: - if (BytesPerWord == 4 and aSize == 8) { + if (TargetBytesPerWord == 4 and aSize == 8) { *thunk = true; } else { *aTypeMask = (1 << RegisterOperand); @@ -3404,15 +3406,15 @@ class MyAssembler: public Assembler { Register stack(rsp); Memory stackLimit(rbx, stackLimitOffsetFromThread); Constant handlerConstant(resolved(&c, handler)); - branchRM(&c, JumpIfGreaterOrEqual, BytesPerWord, &stack, &stackLimit, + branchRM(&c, JumpIfGreaterOrEqual, TargetBytesPerWord, &stack, &stackLimit, &handlerConstant); } virtual void saveFrame(unsigned stackOffset, unsigned) { Register stack(rsp); Memory stackDst(rbx, stackOffset); - apply(Move, BytesPerWord, RegisterOperand, &stack, - BytesPerWord, MemoryOperand, &stackDst); + apply(Move, TargetBytesPerWord, RegisterOperand, &stack, + TargetBytesPerWord, MemoryOperand, &stackDst); } virtual void pushFrame(unsigned argumentCount, ...) { @@ -3430,7 +3432,7 @@ class MyAssembler: public Assembler { = static_cast(va_arg(a, int)); RUNTIME_ARRAY_BODY(arguments)[i].operand = va_arg(a, Operand*); footprint += ceiling - (RUNTIME_ARRAY_BODY(arguments)[i].size, BytesPerWord); + (RUNTIME_ARRAY_BODY(arguments)[i].size, TargetBytesPerWord); } va_end(a); @@ -3448,7 +3450,7 @@ class MyAssembler: public Assembler { RegisterOperand, &dst); } else { - Memory dst(rsp, offset * BytesPerWord); + Memory dst(rsp, offset * TargetBytesPerWord); apply(Move, RUNTIME_ARRAY_BODY(arguments)[i].size, RUNTIME_ARRAY_BODY(arguments)[i].type, @@ -3456,7 +3458,8 @@ class MyAssembler: public Assembler { pad(RUNTIME_ARRAY_BODY(arguments)[i].size), MemoryOperand, &dst); - offset += ceiling(RUNTIME_ARRAY_BODY(arguments)[i].size, BytesPerWord); + offset += ceiling + (RUNTIME_ARRAY_BODY(arguments)[i].size, TargetBytesPerWord); } } } @@ -3466,40 +3469,40 @@ class MyAssembler: public Assembler { if (UseFramePointer) { Register base(rbp); - pushR(&c, BytesPerWord, &base); + pushR(&c, TargetBytesPerWord, &base); - apply(Move, BytesPerWord, RegisterOperand, &stack, - BytesPerWord, RegisterOperand, &base); + apply(Move, TargetBytesPerWord, RegisterOperand, &stack, + TargetBytesPerWord, RegisterOperand, &base); } - Constant footprintConstant(resolved(&c, footprint * BytesPerWord)); - apply(Subtract, BytesPerWord, ConstantOperand, &footprintConstant, - BytesPerWord, RegisterOperand, &stack, - BytesPerWord, RegisterOperand, &stack); + Constant footprintConstant(resolved(&c, footprint * TargetBytesPerWord)); + apply(Subtract, TargetBytesPerWord, ConstantOperand, &footprintConstant, + TargetBytesPerWord, RegisterOperand, &stack, + TargetBytesPerWord, RegisterOperand, &stack); } virtual void adjustFrame(unsigned difference) { Register stack(rsp); - Constant differenceConstant(resolved(&c, difference * BytesPerWord)); - apply(Subtract, BytesPerWord, ConstantOperand, &differenceConstant, - BytesPerWord, RegisterOperand, &stack, - BytesPerWord, RegisterOperand, &stack); + Constant differenceConstant(resolved(&c, difference * TargetBytesPerWord)); + apply(Subtract, TargetBytesPerWord, ConstantOperand, &differenceConstant, + TargetBytesPerWord, RegisterOperand, &stack, + TargetBytesPerWord, RegisterOperand, &stack); } virtual void popFrame(unsigned frameFootprint) { if (UseFramePointer) { Register base(rbp); Register stack(rsp); - apply(Move, BytesPerWord, RegisterOperand, &base, - BytesPerWord, RegisterOperand, &stack); + apply(Move, TargetBytesPerWord, RegisterOperand, &base, + TargetBytesPerWord, RegisterOperand, &stack); - popR(&c, BytesPerWord, &base); + popR(&c, TargetBytesPerWord, &base); } else { Register stack(rsp); - Constant footprint(resolved(&c, frameFootprint * BytesPerWord)); - apply(Add, BytesPerWord, ConstantOperand, &footprint, - BytesPerWord, RegisterOperand, &stack, - BytesPerWord, RegisterOperand, &stack); + Constant footprint(resolved(&c, frameFootprint * TargetBytesPerWord)); + apply(Add, TargetBytesPerWord, ConstantOperand, &footprint, + TargetBytesPerWord, RegisterOperand, &stack, + TargetBytesPerWord, RegisterOperand, &stack); } } @@ -3515,40 +3518,44 @@ class MyAssembler: public Assembler { unsigned baseSize = UseFramePointer ? 1 : 0; Memory returnAddressSrc - (rsp, (frameFootprint + baseSize) * BytesPerWord); - moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &tmp); + (rsp, (frameFootprint + baseSize) * TargetBytesPerWord); + moveMR(&c, TargetBytesPerWord, &returnAddressSrc, TargetBytesPerWord, + &tmp); Memory returnAddressDst - (rsp, (frameFootprint - offset + baseSize) * BytesPerWord); - moveRM(&c, BytesPerWord, &tmp, BytesPerWord, &returnAddressDst); + (rsp, (frameFootprint - offset + baseSize) * TargetBytesPerWord); + moveRM(&c, TargetBytesPerWord, &tmp, TargetBytesPerWord, + &returnAddressDst); c.client->releaseTemporary(tmp.low); if (UseFramePointer) { - Memory baseSrc(rsp, frameFootprint * BytesPerWord); + Memory baseSrc(rsp, frameFootprint * TargetBytesPerWord); Register base(rbp); - moveMR(&c, BytesPerWord, &baseSrc, BytesPerWord, &base); + moveMR(&c, TargetBytesPerWord, &baseSrc, TargetBytesPerWord, &base); } Register stack(rsp); Constant footprint - (resolved(&c, (frameFootprint - offset + baseSize) * BytesPerWord)); - addCR(&c, BytesPerWord, &footprint, BytesPerWord, &stack); + (resolved + (&c, (frameFootprint - offset + baseSize) * TargetBytesPerWord)); + + addCR(&c, TargetBytesPerWord, &footprint, TargetBytesPerWord, &stack); if (returnAddressSurrogate != NoRegister) { assert(&c, offset > 0); Register ras(returnAddressSurrogate); - Memory dst(rsp, offset * BytesPerWord); - moveRM(&c, BytesPerWord, &ras, BytesPerWord, &dst); + Memory dst(rsp, offset * TargetBytesPerWord); + moveRM(&c, TargetBytesPerWord, &ras, TargetBytesPerWord, &dst); } if (framePointerSurrogate != NoRegister) { assert(&c, offset > 0); Register fps(framePointerSurrogate); - Memory dst(rsp, (offset - 1) * BytesPerWord); - moveRM(&c, BytesPerWord, &fps, BytesPerWord, &dst); + Memory dst(rsp, (offset - 1) * TargetBytesPerWord); + moveRM(&c, TargetBytesPerWord, &fps, TargetBytesPerWord, &dst); } } else { popFrame(frameFootprint); @@ -3568,15 +3575,15 @@ class MyAssembler: public Assembler { if (TailCalls and argumentFootprint > StackAlignmentInWords) { Register returnAddress(rcx); - popR(&c, BytesPerWord, &returnAddress); + popR(&c, TargetBytesPerWord, &returnAddress); Register stack(rsp); Constant adjustment (resolved(&c, (argumentFootprint - StackAlignmentInWords) - * BytesPerWord)); - addCR(&c, BytesPerWord, &adjustment, BytesPerWord, &stack); + * TargetBytesPerWord)); + addCR(&c, TargetBytesPerWord, &adjustment, TargetBytesPerWord, &stack); - jumpR(&c, BytesPerWord, &returnAddress); + jumpR(&c, TargetBytesPerWord, &returnAddress); } else { return_(&c); } @@ -3588,13 +3595,13 @@ class MyAssembler: public Assembler { popFrame(frameFootprint); Register returnAddress(rcx); - popR(&c, BytesPerWord, &returnAddress); + popR(&c, TargetBytesPerWord, &returnAddress); Register stack(rsp); Memory stackSrc(rbx, stackOffsetFromThread); - moveMR(&c, BytesPerWord, &stackSrc, BytesPerWord, &stack); + moveMR(&c, TargetBytesPerWord, &stackSrc, TargetBytesPerWord, &stack); - jumpR(&c, BytesPerWord, &returnAddress); + jumpR(&c, TargetBytesPerWord, &returnAddress); } virtual void apply(Operation op) { @@ -3624,7 +3631,7 @@ class MyAssembler: public Assembler { { if (isBranch(op)) { assert(&c, aSize == bSize); - assert(&c, cSize == BytesPerWord); + assert(&c, cSize == TargetBytesPerWord); assert(&c, cType == ConstantOperand); arch_->c.branchOperations[branchIndex(&(arch_->c), aType, bType)] From 67300c229adf12169f2b7636f46bc0def77003ed Mon Sep 17 00:00:00 2001 From: Joel Dice Date: Wed, 31 Aug 2011 21:15:41 -0600 Subject: [PATCH 2/5] change local variable name to avoid shadowing another variable --- src/machine.cpp | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/machine.cpp b/src/machine.cpp index 0219894f77..25054e33da 100644 --- a/src/machine.cpp +++ b/src/machine.cpp @@ -1452,15 +1452,15 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool) unsigned attributeCount = s.read2(); for (unsigned j = 0; j < attributeCount; ++j) { - object name = singletonObject(t, pool, s.read2() - 1); + object attributeName = singletonObject(t, pool, s.read2() - 1); unsigned length = s.read4(); if (vm::strcmp(reinterpret_cast("Code"), - &byteArrayBody(t, name, 0)) == 0) + &byteArrayBody(t, attributeName, 0)) == 0) { code = parseCode(t, s, pool); } else if (vm::strcmp(reinterpret_cast("Exceptions"), - &byteArrayBody(t, name, 0)) == 0) + &byteArrayBody(t, attributeName, 0)) == 0) { if (addendum == 0) { addendum = makeMethodAddendum(t, pool, 0, 0, 0, 0); @@ -1473,7 +1473,7 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool) set(t, addendum, MethodAddendumExceptionTable, body); } else if (vm::strcmp(reinterpret_cast ("AnnotationDefault"), - &byteArrayBody(t, name, 0)) == 0) + &byteArrayBody(t, attributeName, 0)) == 0) { if (addendum == 0) { addendum = makeMethodAddendum(t, pool, 0, 0, 0, 0); @@ -1485,7 +1485,7 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool) set(t, addendum, MethodAddendumAnnotationDefault, body); } else if (vm::strcmp(reinterpret_cast("Signature"), - &byteArrayBody(t, name, 0)) == 0) + &byteArrayBody(t, attributeName, 0)) == 0) { if (addendum == 0) { addendum = makeMethodAddendum(t, pool, 0, 0, 0, 0); @@ -1495,7 +1495,7 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool) singletonObject(t, pool, s.read2() - 1)); } else if (vm::strcmp(reinterpret_cast ("RuntimeVisibleAnnotations"), - &byteArrayBody(t, name, 0)) == 0) + &byteArrayBody(t, attributeName, 0)) == 0) { if (addendum == 0) { addendum = makeMethodAddendum(t, pool, 0, 0, 0, 0); From 5d55d61c17f53f0f2ad1080e0548db8b10b8b026 Mon Sep 17 00:00:00 2001 From: Joel Dice Date: Wed, 31 Aug 2011 21:16:22 -0600 Subject: [PATCH 3/5] fix use of uninitalized memory in jnienv.cpp --- src/jnienv.cpp | 14 +++++++++----- 1 file changed, 9 insertions(+), 5 deletions(-) diff --git a/src/jnienv.cpp b/src/jnienv.cpp index a18e8433c0..4d4518febd 100644 --- a/src/jnienv.cpp +++ b/src/jnienv.cpp @@ -3324,11 +3324,15 @@ JNI_CreateJavaVM(Machine** m, Thread** t, void* args) unsigned bootClasspathBufferSize = bcppl + bcpl + bcpal + 3; RUNTIME_ARRAY(char, bootClasspathBuffer, bootClasspathBufferSize); char* bootClasspathPointer = RUNTIME_ARRAY_BODY(bootClasspathBuffer); - local::append(&bootClasspathPointer, bootClasspathPrepend, bcppl, - bcpl + bcpal ? PATH_SEPARATOR : 0); - local::append(&bootClasspathPointer, bootClasspath, bcpl, - bcpal ? PATH_SEPARATOR : 0); - local::append(&bootClasspathPointer, bootClasspathAppend, bcpal, 0); + if (bootClasspathBufferSize > 3) { + local::append(&bootClasspathPointer, bootClasspathPrepend, bcppl, + bcpl + bcpal ? PATH_SEPARATOR : 0); + local::append(&bootClasspathPointer, bootClasspath, bcpl, + bcpal ? PATH_SEPARATOR : 0); + local::append(&bootClasspathPointer, bootClasspathAppend, bcpal, 0); + } else { + *RUNTIME_ARRAY_BODY(bootClasspathBuffer) = 0; + } Finder* bf = makeFinder (s, h, RUNTIME_ARRAY_BODY(bootClasspathBuffer), bootLibrary); From e505cbe99d097e726ff3c1c22212af3b015d4ec1 Mon Sep 17 00:00:00 2001 From: Joel Dice Date: Wed, 31 Aug 2011 21:18:00 -0600 Subject: [PATCH 4/5] more progress towards cross-architecture bootimage builds This commit fixes a lot of bugs. All tests are now pass for Linux x86_64 to Linux i386 cross builds. --- makefile | 23 +- src/bootimage-template.cpp | 30 ++ src/bootimage.cpp | 639 +++++++++++++++++++++++++++---------- src/bootimage.h | 102 +++--- src/common.h | 4 + src/compile.cpp | 222 ++++++++----- src/compiler.cpp | 10 +- src/heap.cpp | 2 +- src/interpret.cpp | 6 +- src/processor.h | 6 +- src/target.h | 43 ++- src/type-generator.cpp | 3 + src/vector.h | 5 + src/x86.cpp | 12 +- 14 files changed, 779 insertions(+), 328 deletions(-) create mode 100644 src/bootimage-template.cpp diff --git a/makefile b/makefile index a3a6519c2b..e8645506f4 100755 --- a/makefile +++ b/makefile @@ -165,12 +165,13 @@ rdynamic = -rdynamic warnings = -Wall -Wextra -Werror -Wunused-parameter -Winit-self \ -Wno-non-virtual-dtor +target-cflags = -DTARGET_BYTES_PER_WORD=$(pointer-size) + common-cflags = $(warnings) -fno-rtti -fno-exceptions \ "-I$(JAVA_HOME)/include" -idirafter $(src) -I$(build) $(classpath-cflags) \ -D__STDC_LIMIT_MACROS -D_JNI_IMPLEMENTATION_ -DAVIAN_VERSION=\"$(version)\" \ -DUSE_ATOMIC_OPERATIONS -DAVIAN_JAVA_HOME=\"$(javahome)\" \ - -DAVIAN_EMBED_PREFIX=\"$(embed-prefix)\" \ - -DTARGET_BYTES_PER_WORD=$(pointer-size) + -DAVIAN_EMBED_PREFIX=\"$(embed-prefix)\" $(target-cflags) ifneq (,$(filter i386 x86_64,$(arch))) ifeq ($(use-frame-pointer),true) @@ -209,9 +210,11 @@ shared = -shared openjdk-extra-cflags = -fvisibility=hidden +bootimage-cflags = -DTARGET_BYTES_PER_WORD=$(pointer-size) + ifeq ($(build-arch),powerpc) ifneq ($(arch),$(build-arch)) - cflags += -DTARGET_OPPOSITE_ENDIAN + bootimage-cflags += -DTARGET_OPPOSITE_ENDIAN endif endif @@ -223,7 +226,7 @@ ifeq ($(arch),powerpc) pointer-size = 4 ifneq ($(arch),$(build-arch)) - cflags += -DTARGET_OPPOSITE_ENDIAN + bootimage-cflags += -DTARGET_OPPOSITE_ENDIAN endif ifneq ($(platform),darwin) @@ -264,7 +267,7 @@ endif ifeq ($(platform),linux) - cflags += -DTARGET_PLATFORM_LINUX + bootimage-cflags += -DTARGET_PLATFORM_LINUX endif ifeq ($(build-platform),darwin) @@ -274,7 +277,7 @@ ifeq ($(build-platform),darwin) endif ifeq ($(platform),darwin) - cflags += -DTARGET_PLATFORM_DARWIN + bootimage-cflags += -DTARGET_PLATFORM_DARWIN ifeq (${OSX_SDK_SYSROOT},) OSX_SDK_SYSROOT = 10.4u @@ -350,7 +353,7 @@ ifeq ($(platform),darwin) endif ifeq ($(platform),windows) - cflags += -DTARGET_PLATFORM_WINDOWS + bootimage-cflags += -DTARGET_PLATFORM_WINDOWS inc = "$(root)/win32/include" lib = "$(root)/win32/lib" @@ -520,10 +523,12 @@ vm-sources = \ vm-asm-sources = $(src)/$(asm).S +target-asm = $(asm) + ifeq ($(process),compile) vm-sources += \ $(src)/compiler.cpp \ - $(src)/$(asm).cpp + $(src)/$(target-asm).cpp vm-asm-sources += $(src)/compile-$(asm).S endif @@ -902,6 +907,8 @@ $(bootimage-generator): openjdk-src=$(openjdk-src) \ bootimage-generator= \ build-bootimage-generator=$(bootimage-generator) \ + target-cflags="$(bootimage-cflags)" \ + target-asm=$(asm) \ $(bootimage-generator) $(build-bootimage-generator): \ diff --git a/src/bootimage-template.cpp b/src/bootimage-template.cpp new file mode 100644 index 0000000000..d96a586fcd --- /dev/null +++ b/src/bootimage-template.cpp @@ -0,0 +1,30 @@ +const unsigned NAME(BootMask) = (~static_cast(0)) + / NAME(BytesPerWord); + +const unsigned NAME(BootShift) = 32 - log(NAME(BytesPerWord)); + +const unsigned NAME(BootFlatConstant) = 1 << NAME(BootShift); +const unsigned NAME(BootHeapOffset) = 1 << (NAME(BootShift) + 1); + +inline unsigned +LABEL(codeMapSize)(unsigned codeSize) +{ + return ceiling(codeSize, TargetBitsPerWord) * TargetBytesPerWord; +} + +inline unsigned +LABEL(heapMapSize)(unsigned heapSize) +{ + return ceiling(heapSize, TargetBitsPerWord * TargetBytesPerWord) + * TargetBytesPerWord; +} + +inline object +LABEL(bootObject)(LABEL(uintptr_t)* heap, unsigned offset) +{ + if (offset) { + return reinterpret_cast(heap + offset - 1); + } else { + return 0; + } +} diff --git a/src/bootimage.cpp b/src/bootimage.cpp index 0626eb5c77..99d7a40086 100644 --- a/src/bootimage.cpp +++ b/src/bootimage.cpp @@ -8,7 +8,6 @@ There is NO WARRANTY for this software. See license.txt for details. */ -#include "bootimage.h" #include "heap.h" #include "heapwalk.h" #include "common.h" @@ -40,6 +39,7 @@ const bool DebugNativeTarget = false; enum Type { Type_none, Type_object, + Type_object_nogc, Type_int8_t, Type_uint8_t, Type_int16_t, @@ -60,13 +60,19 @@ enum Type { class Field { public: - Field(Type type, unsigned offset, unsigned targetOffset): - type(type), offset(offset), targetOffset(targetOffset) + Field() { } + + Field(Type type, unsigned buildOffset, unsigned buildSize, + unsigned targetOffset, unsigned targetSize): + type(type), buildOffset(buildOffset), buildSize(buildSize), + targetOffset(targetOffset), targetSize(targetSize) { } Type type; - unsigned offset; + unsigned buildOffset; + unsigned buildSize; unsigned targetOffset; + unsigned targetSize; }; class TypeMap { @@ -155,6 +161,108 @@ endsWith(const char* suffix, const char* s, unsigned length) and memcmp(suffix, s + (length - suffixLength), suffixLength) == 0; } +object +getNonStaticFields(Thread* t, object typeMaps, object c, object fields, + unsigned* count, object* array) +{ + PROTECT(t, typeMaps); + PROTECT(t, c); + PROTECT(t, fields); + + *array = hashMapFind(t, typeMaps, c, objectHash, objectEqual); + + if (*array) { + *count += reinterpret_cast(&byteArrayBody(t, *array, 0)) + ->fixedFieldCount; + } else { + if (classSuper(t, c)) { + fields = getNonStaticFields + (t, typeMaps, classSuper(t, c), fields, count, array); + } + + if (classFieldTable(t, c)) { + for (unsigned i = 0; i < arrayLength(t, classFieldTable(t, c)); ++i) { + object field = arrayBody(t, classFieldTable(t, c), i); + + if ((fieldFlags(t, field) & ACC_STATIC) == 0) { + ++ (*count); + fields = vectorAppend(t, fields, field); + } + } + } + } + + return vectorAppend(t, fields, 0); +} + +object +allFields(Thread* t, object typeMaps, object c, unsigned* count, object* array) +{ + PROTECT(t, typeMaps); + PROTECT(t, c); + + object fields = makeVector(t, 0, 0); + PROTECT(t, fields); + + *array = hashMapFind(t, typeMaps, c, objectHash, objectEqual); + + bool includeMembers; + if (*array) { + includeMembers = false; + *count += reinterpret_cast(&byteArrayBody(t, *array, 0)) + ->fixedFieldCount; + } else if (classSuper(t, c)) { + includeMembers = true; + fields = getNonStaticFields + (t, typeMaps, classSuper(t, c), fields, count, array); + } + + if (classFieldTable(t, c)) { + for (unsigned i = 0; i < arrayLength(t, classFieldTable(t, c)); ++i) { + object field = arrayBody(t, classFieldTable(t, c), i); + + if (includeMembers or (fieldFlags(t, field) & ACC_STATIC)) { + ++ (*count); + fields = vectorAppend(t, fields, field); + } + } + } + + return fields; +} + +TypeMap* +classTypeMap(Thread* t, object typeMaps, object p) +{ + return reinterpret_cast + (&byteArrayBody + (t, hashMapFind(t, typeMaps, p, objectHash, objectEqual), 0)); +} + +TypeMap* +typeMap(Thread* t, object typeMaps, object p) +{ + return reinterpret_cast + (&byteArrayBody + (t, objectClass(t, p) == type(t, Machine::SingletonType) + ? hashMapFind(t, typeMaps, p, objectHash, objectEqual) + : hashMapFind(t, typeMaps, objectClass(t, p), objectHash, objectEqual), + 0)); +} + +unsigned +targetFieldOffset(Thread* t, object typeMaps, object field) +{ + // if (strcmp(reinterpret_cast + // (&byteArrayBody(t, className(t, fieldClass(t, field)), 0)), + // "java/lang/Throwable") == 0) trap(); + + return ((fieldFlags(t, field) & ACC_STATIC) + ? typeMap(t, typeMaps, classStaticTable(t, fieldClass(t, field))) + : classTypeMap(t, typeMaps, fieldClass(t, field))) + ->targetFixedOffsets()[fieldOffset(t, field)]; +} + object makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, uintptr_t* codeMap, const char* className, @@ -168,8 +276,22 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, object calls = 0; PROTECT(t, calls); + object methods = 0; + PROTECT(t, methods); + DelayedPromise* addresses = 0; + class MyOffsetResolver: public OffsetResolver { + public: + MyOffsetResolver(object* typeMaps): typeMaps(typeMaps) { } + + virtual unsigned fieldOffset(Thread* t, object field) { + return targetFieldOffset(t, *typeMaps, field); + } + + object* typeMaps; + } resolver(&typeMaps); + Finder* finder = static_cast (systemClassLoaderFinder(t, root(t, Machine::BootLoader))); @@ -272,7 +394,8 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, = i * TargetBytesPerWord; new (map->fixedFields() + i) Field - (types[i], i * BytesPerWord, i * TargetBytesPerWord); + (types[i], i * BytesPerWord, BytesPerWord, + i * TargetBytesPerWord, TargetBytesPerWord); } hashMapInsert @@ -281,101 +404,135 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, objectHash); } } + + // if (strcmp(name, "java/lang/System$Property.class") == 0) trap(); - if (classFieldTable(t, c)) { - unsigned count = arrayLength(t, classFieldTable(t, c)); + { object array = 0; + PROTECT(t, array); - Type memberTypes[count + 1]; - memberTypes[0] = Type_object; - unsigned buildMemberOffsets[count + 1]; - buildMemberOffsets[0] = 0; - unsigned targetMemberOffsets[count + 1]; - targetMemberOffsets[0] = 0; - unsigned memberIndex = 1; - unsigned buildMemberOffset = BytesPerWord; - unsigned targetMemberOffset = TargetBytesPerWord; + unsigned count = 0; + object fields = allFields(t, typeMaps, c, &count, &array); + PROTECT(t, fields); + + Field memberFields[count + 1]; + + unsigned memberIndex; + unsigned buildMemberOffset; + unsigned targetMemberOffset; + + if (array) { + memberIndex = 0; + buildMemberOffset = 0; + targetMemberOffset = 0; + + TypeMap* map = reinterpret_cast + (&byteArrayBody(t, array, 0)); + + for (unsigned j = 0; j < map->fixedFieldCount; ++j) { + Field* f = map->fixedFields() + j; + + memberFields[memberIndex] = *f; + + targetMemberOffset += f->targetSize; + + ++ memberIndex; + } + } else { + new (memberFields) Field + (Type_object, 0, BytesPerWord, 0, TargetBytesPerWord); + + memberIndex = 1; + buildMemberOffset = BytesPerWord; + targetMemberOffset = TargetBytesPerWord; + } + + Field staticFields[count + 2]; + + new (staticFields) Field + (Type_object, 0, BytesPerWord, 0, TargetBytesPerWord); + + new (staticFields + 1) Field + (Type_intptr_t, BytesPerWord, BytesPerWord, TargetBytesPerWord, + TargetBytesPerWord); - Type staticTypes[count + 2]; - staticTypes[0] = Type_object; - staticTypes[1] = Type_intptr_t; - unsigned buildStaticOffsets[count + 2]; - buildStaticOffsets[0] = 0; - buildStaticOffsets[1] = BytesPerWord; - unsigned targetStaticOffsets[count + 2]; - targetStaticOffsets[0] = 0; - targetStaticOffsets[1] = TargetBytesPerWord; unsigned staticIndex = 2; unsigned buildStaticOffset = BytesPerWord * 2; unsigned targetStaticOffset = TargetBytesPerWord * 2; - for (unsigned i = 0; i < count; ++i) { - object field = arrayBody(t, classFieldTable(t, c), i); - unsigned size = fieldSize(t, fieldCode(t, field)); + for (unsigned i = 0; i < vectorSize(t, fields); ++i) { + object field = vectorBody(t, fields, i); + if (field) { + unsigned buildSize = fieldSize(t, fieldCode(t, field)); + unsigned targetSize = buildSize; - Type type; - switch (fieldCode(t, field)) { - case ObjectField: - type = Type_object; - size = TargetBytesPerWord; - break; + Type type; + switch (fieldCode(t, field)) { + case ObjectField: + type = Type_object; + targetSize = TargetBytesPerWord; + break; - case ByteField: - case BooleanField: - type = Type_int8_t; - break; + case ByteField: + case BooleanField: + type = Type_int8_t; + break; - case CharField: - case ShortField: - type = Type_int8_t; - break; + case CharField: + case ShortField: + type = Type_int8_t; + break; - case FloatField: - case IntField: - type = Type_int32_t; - break; + case FloatField: + case IntField: + type = Type_int32_t; + break; - case LongField: - case DoubleField: - type = Type_int64_t; - break; + case LongField: + case DoubleField: + type = Type_int64_t; + break; - default: abort(t); - } - - if (fieldFlags(t, field) & ACC_STATIC) { - staticTypes[staticIndex] = type; - - while (targetStaticOffset % size) { - ++ targetStaticOffset; + default: abort(t); } - targetStaticOffsets[staticIndex] = targetStaticOffset; + if (fieldFlags(t, field) & ACC_STATIC) { + while (targetStaticOffset % targetSize) { + ++ targetStaticOffset; + } - targetStaticOffset += size; + buildStaticOffset = fieldOffset(t, field); - buildStaticOffset = fieldOffset(t, field); - buildStaticOffsets[staticIndex] = buildStaticOffset; + new (staticFields + staticIndex) Field + (type, buildStaticOffset, buildSize, targetStaticOffset, + targetSize); - ++ staticIndex; + targetStaticOffset += targetSize; + + ++ staticIndex; + } else { + while (targetMemberOffset % targetSize) { + ++ targetMemberOffset; + } + + buildMemberOffset = fieldOffset(t, field); + + new (memberFields + memberIndex) Field + (type, buildMemberOffset, buildSize, targetMemberOffset, + targetSize); + + targetMemberOffset += targetSize; + + ++ memberIndex; + } } else { - memberTypes[memberIndex] = type; - - while (targetMemberOffset % size) { - ++ targetMemberOffset; - } - - targetMemberOffsets[memberIndex] = targetMemberOffset; - - targetMemberOffset += size; - - buildMemberOffset = fieldOffset(t, field); - buildMemberOffsets[memberIndex] = buildMemberOffset; - - ++ memberIndex; + targetMemberOffset = pad(targetMemberOffset, TargetBytesPerWord); } } - { object array = makeByteArray + // if (strcmp(name, "avian/VMClass.class") == 0) trap(); + + if (hashMapFind(t, typeMaps, c, objectHash, objectEqual) == 0) { + object array = makeByteArray (t, TypeMap::sizeInBytes (ceiling(classFixedSize(t, c), BytesPerWord), memberIndex)); @@ -384,14 +541,14 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, ceiling(targetMemberOffset, TargetBytesPerWord), memberIndex); for (unsigned i = 0; i < memberIndex; ++i) { - expect(t, buildMemberOffsets[i] + Field* f = memberFields + i; + + expect(t, f->buildOffset < map->buildFixedSizeInWords * BytesPerWord); - map->targetFixedOffsets()[buildMemberOffsets[i]] - = targetMemberOffsets[i]; + map->targetFixedOffsets()[f->buildOffset] = f->targetOffset; - new (map->fixedFields() + i) Field - (memberTypes[i], buildMemberOffsets[i], targetMemberOffsets[i]); + map->fixedFields()[i] = *f; } hashMapInsert(t, typeMaps, c, array, objectHash); @@ -408,20 +565,36 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, TypeMap::SingletonKind); for (unsigned i = 0; i < staticIndex; ++i) { - expect(t, buildStaticOffsets[i] + Field* f = staticFields + i; + + expect(t, f->buildOffset < map->buildFixedSizeInWords * BytesPerWord); - map->targetFixedOffsets()[buildStaticOffsets[i]] - = targetStaticOffsets[i]; + map->targetFixedOffsets()[f->buildOffset] = f->targetOffset; - new (map->fixedFields() + i) Field - (staticTypes[i], buildStaticOffsets[i], targetStaticOffsets[i]); + map->fixedFields()[i] = *f; } hashMapInsert (t, typeMaps, classStaticTable(t, c), array, objectHash); } } + } + } + + for (Finder::Iterator it(finder); it.hasMore();) { + unsigned nameSize = 0; + const char* name = it.next(&nameSize); + + if (endsWith(".class", name, nameSize) + and (className == 0 or strncmp(name, className, nameSize - 6) == 0)) + { + // fprintf(stderr, "%.*s\n", nameSize - 6, name); + object c = resolveSystemClass + (t, root(t, Machine::BootLoader), + makeByteArray(t, "%.*s", nameSize - 6, name), true); + + PROTECT(t, c); if (classMethodTable(t, c)) { for (unsigned i = 0; i < arrayLength(t, classMethodTable(t, c)); ++i) { @@ -444,7 +617,11 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, PROTECT(t, method); t->m->processor->compileMethod - (t, zone, &constants, &calls, &addresses, method); + (t, zone, &constants, &calls, &addresses, method, &resolver); + + if (methodCode(t, method)) { + methods = makePair(t, method, methods); + } } object addendum = methodAddendum(t, method); @@ -499,7 +676,7 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, (reinterpret_cast(code), &location); target_uintptr_t offset = value - code; if (flat) { - offset |= BootFlatConstant; + offset |= TargetBootFlatConstant; } memcpy(location, &offset, TargetBytesPerWord); @@ -510,6 +687,13 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code, - reinterpret_cast(code)); } + for (; methods; methods = pairSecond(t, methods)) { + codeCompiled(t, methodCode(t, pairFirst(t, methods))) + -= reinterpret_cast(code); + } + + t->m->processor->normalizeVirtualThunks(t); + return constants; } @@ -541,17 +725,6 @@ visitRoots(Thread* t, BootImage* image, HeapWalker* w, object constants) } } -TypeMap* -typeMap(Thread* t, object typeMaps, object p) -{ - return reinterpret_cast - (&byteArrayBody - (t, objectClass(t, p) == type(t, Machine::SingletonType) - ? hashMapFind(t, typeMaps, p, objectHash, objectEqual) - : hashMapFind(t, typeMaps, objectClass(t, p), objectHash, objectEqual), - 0)); -} - unsigned targetOffset(Thread* t, object typeMaps, object p, unsigned offset) { @@ -602,6 +775,36 @@ targetSize(Thread* t, object typeMaps, object p) } } +unsigned +objectMaskCount(TypeMap* map) +{ + unsigned count = map->targetFixedSizeInWords; + + if (map->targetArrayElementSizeInBytes) { + ++ count; + } + + return count; +} + +unsigned +targetSize(Thread* t, object typeMaps, object referer, unsigned refererOffset, + object p) +{ + if (referer + and objectClass(t, referer) == type(t, Machine::ClassType) + and (refererOffset * BytesPerWord) == ClassObjectMask) + { + return (TargetBytesPerWord * 2) + + pad + (ceiling + (objectMaskCount + (classTypeMap(t, typeMaps, referer)), 32) * 4, TargetBytesPerWord); + } else { + return targetSize(t, typeMaps, p); + } +} + void copy(Thread* t, uint8_t* src, uint8_t* dst, Type type) { @@ -674,6 +877,7 @@ nonObjectsEqual(uint8_t* src, uint8_t* dst, Type type) return memcmp(dst, src, BytesPerWord) == 0; case Type_object: + case Type_object_nogc: return true; default: abort(); @@ -686,7 +890,7 @@ nonObjectsEqual(TypeMap* map, uint8_t* src, uint8_t* dst) for (unsigned i = 0; i < map->fixedFieldCount; ++i) { Field* field = map->fixedFields() + i; if (not nonObjectsEqual - (src + field->offset, dst + field->targetOffset, field->type)) + (src + field->buildOffset, dst + field->targetOffset, field->type)) { return false; } @@ -720,7 +924,7 @@ copy(Thread* t, object typeMaps, object p, uint8_t* dst) for (unsigned i = 0; i < map->fixedFieldCount; ++i) { Field* field = map->fixedFields() + i; if (field->type > Type_array) abort(t); - copy(t, src + field->offset, dst + field->targetOffset, field->type); + copy(t, src + field->buildOffset, dst + field->targetOffset, field->type); } if (map->targetArrayElementSizeInBytes) { @@ -732,16 +936,59 @@ copy(Thread* t, object typeMaps, object p, uint8_t* dst) dst + (map->targetFixedSizeInWords * TargetBytesPerWord) + (i * map->targetArrayElementSizeInBytes), map->arrayElementType); } + + if (objectClass(t, p) == type(t, Machine::ClassType)) { + uint16_t fixedSize; + uint8_t arrayElementSize; + object array = hashMapFind(t, typeMaps, p, objectHash, objectEqual); + + if (array) { + TypeMap* classMap = reinterpret_cast + (&byteArrayBody(t, array, 0)); + + fixedSize = TARGET_V2 + (classMap->targetFixedSizeInWords * TargetBytesPerWord); + + arrayElementSize = classMap->targetArrayElementSizeInBytes; + } else if (classFixedSize(t, p) == BytesPerWord * 2 + and classArrayElementSize(t, p) == BytesPerWord) + { + fixedSize = TARGET_V2(TargetBytesPerWord * 2); + + arrayElementSize = TargetBytesPerWord; + } else { + fixedSize = 0; + arrayElementSize = 0; + } + + if (fixedSize) { + memcpy(dst + TargetClassFixedSize, &fixedSize, 2); + + memcpy(dst + TargetClassArrayElementSize, &arrayElementSize, 1); + } + + // if (strcmp("vm::lineNumberTable", + // reinterpret_cast(&byteArrayBody(t, className(t, p), 0))) == 0) trap(); + } } else { switch (map->kind) { case TypeMap::NormalKind: + if (objectClass(t, p) == type(t, Machine::FieldType)) { + uint16_t offset = TARGET_V2(targetFieldOffset(t, typeMaps, p)); + memcpy(dst + TargetFieldOffset, &offset, 2); + } break; case TypeMap::SingletonKind: { + unsigned maskSize = singletonMaskSize + (map->targetFixedSizeInWords - 2, TargetBitsPerWord); + + target_uintptr_t targetLength = TARGET_VW + (map->targetFixedSizeInWords - 2 + maskSize); + memcpy(dst + TargetBytesPerWord, &targetLength, TargetBytesPerWord); + uint8_t* mask = dst + (map->targetFixedSizeInWords * TargetBytesPerWord); - memset(mask, 0, singletonMaskSize - (map->targetFixedSizeInWords - 2, TargetBitsPerWord) - * TargetBytesPerWord); + memset(mask, 0, maskSize * TargetBytesPerWord); for (unsigned i = 0; i < map->fixedFieldCount; ++i) { Field* field = map->fixedFields() + i; @@ -766,6 +1013,13 @@ copy(Thread* t, object typeMaps, object p, uint8_t* dst) unsigned poolMaskSize = vm::poolMaskSize (map->targetFixedSizeInWords - 2, TargetBitsPerWord); + unsigned objectMaskSize = singletonMaskSize + (map->targetFixedSizeInWords - 2 + poolMaskSize, TargetBitsPerWord); + + target_uintptr_t targetLength = TARGET_VW + (map->targetFixedSizeInWords - 2 + poolMaskSize + objectMaskSize); + memcpy(dst + TargetBytesPerWord, &targetLength, TargetBytesPerWord); + uint8_t* poolMask = dst + (map->targetFixedSizeInWords * TargetBytesPerWord); @@ -774,9 +1028,7 @@ copy(Thread* t, object typeMaps, object p, uint8_t* dst) uint8_t* objectMask = dst + ((map->targetFixedSizeInWords + poolMaskSize) * TargetBytesPerWord); - memset(objectMask, 0, singletonMaskSize - (map->targetFixedSizeInWords - 2 + poolMaskSize, - TargetBitsPerWord) * TargetBytesPerWord); + memset(objectMask, 0, objectMaskSize * TargetBytesPerWord); for (unsigned i = 0; i < map->fixedFieldCount; ++i) { Field* field = map->fixedFields() + i; @@ -812,21 +1064,64 @@ copy(Thread* t, object typeMaps, object p, uint8_t* dst) default: abort(t); } } +} + +void +copy(Thread* t, object typeMaps, object referer, unsigned refererOffset, + object p, uint8_t* dst) +{ + if (referer + and objectClass(t, referer) == type(t, Machine::ClassType) + and (refererOffset * BytesPerWord) == ClassObjectMask) + { + TypeMap* map = classTypeMap(t, typeMaps, referer); + + memset(dst, 0, TargetBytesPerWord); + + unsigned length = ceiling(objectMaskCount(map), 32); + + target_uintptr_t targetLength = TARGET_VW(length); + + memcpy(dst + TargetBytesPerWord, &targetLength, TargetBytesPerWord); + + memset(dst + (TargetBytesPerWord * 2), 0, length * 4); + + for (unsigned i = 0; i < map->fixedFieldCount; ++i) { + Field* field = map->fixedFields() + i; + if (field->type == Type_object) { + unsigned offset = field->targetOffset / TargetBytesPerWord; + reinterpret_cast(dst + (TargetBytesPerWord * 2)) + [offset / 32] |= static_cast(1) << (offset % 32); + } + } + + if (map->targetArrayElementSizeInBytes + and map->arrayElementType == Type_object) + { + unsigned offset = map->targetFixedSizeInWords; + reinterpret_cast(dst + (TargetBytesPerWord * 2)) + [offset / 32] |= static_cast(1) << (offset % 32); + } + } else { + copy(t, typeMaps, p, dst); + } if (DebugNativeTarget) { expect(t, targetSize(t, typeMaps, p) == baseSize(t, p, objectClass(t, p))); - expect(t, nonObjectsEqual(map, src, dst)); + expect(t, nonObjectsEqual + (typeMap(t, typeMaps, p), reinterpret_cast(p), dst)); } } HeapWalker* -makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, - unsigned capacity, object constants, object typeMaps) +makeHeapImage(Thread* t, BootImage* image, target_uintptr_t* heap, + target_uintptr_t* map, unsigned capacity, object constants, + object typeMaps) { class Visitor: public HeapVisitor { public: - Visitor(Thread* t, object typeMaps, uintptr_t* heap, - uintptr_t* map, unsigned capacity): + Visitor(Thread* t, object typeMaps, target_uintptr_t* heap, + target_uintptr_t* map, unsigned capacity): t(t), typeMaps(typeMaps), currentObject(0), currentNumber(0), currentOffset(0), heap(heap), map(map), position(0), capacity(capacity) { } @@ -845,8 +1140,8 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, (t, typeMaps, currentObject, currentOffset * BytesPerWord) / TargetBytesPerWord); - unsigned mark = heap[offset] & (~PointerMask); - unsigned value = number | (mark << BootShift); + unsigned mark = heap[offset] & (~TargetPointerMask); + unsigned value = number | (mark << TargetBootShift); if (value) markBit(map, offset); @@ -860,7 +1155,8 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, virtual unsigned visitNew(object p) { if (p) { - unsigned size = targetSize(t, typeMaps, p); + unsigned size = targetSize + (t, typeMaps, currentObject, currentOffset, p); unsigned number; if ((currentObject @@ -877,7 +1173,7 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, target_uintptr_t* dst = heap + position + TargetFixieSizeInWords; - unsigned maskSize = ceiling(size, TargetBytesPerWord); + unsigned maskSize = ceiling(size, TargetBitsPerWord); unsigned total = TargetFixieSizeInWords + size + maskSize; @@ -897,7 +1193,8 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, memcpy(reinterpret_cast(heap + position) + TargetFixieSize, &targetSize, 4); - copy(t, typeMaps, p, reinterpret_cast(dst)); + copy(t, typeMaps, currentObject, currentOffset, p, + reinterpret_cast(dst)); dst[0] |= FixedMark; @@ -909,7 +1206,8 @@ makeHeapImage(Thread* t, BootImage* image, uintptr_t* heap, uintptr_t* map, } else { expect(t, position + size < capacity); - copy(t, typeMaps, p, reinterpret_cast(heap + position)); + copy(t, typeMaps, currentObject, currentOffset, p, + reinterpret_cast(heap + position)); number = position + 1; position += size; @@ -970,15 +1268,21 @@ updateConstants(Thread* t, object constants, uint8_t* code, uintptr_t* codeMap, { void* location; bool flat = pl->resolve(0, &location); - target_uintptr_t offset = target | BootHeapOffset; + target_uintptr_t offset = target | TargetBootHeapOffset; if (flat) { - offset |= BootFlatConstant; + offset |= TargetBootFlatConstant; } memcpy(location, &offset, TargetBytesPerWord); expect(t, reinterpret_cast(location) >= reinterpret_cast(code)); + // fprintf(stderr, "mark constant %d %d\n", + // static_cast + // (reinterpret_cast(location) + // - reinterpret_cast(code)), + // static_cast(offset)); + markBit(codeMap, reinterpret_cast(location) - reinterpret_cast(code)); } @@ -1014,12 +1318,6 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, typeMaps = makeHashMap(t, 0, 0); PROTECT(t, typeMaps); - constants = makeCodeImage - (t, &zone, image, code, codeMap, className, methodName, methodSpec, - typeMaps); - - PROTECT(t, constants); - #include "type-maps.cpp" for (unsigned i = 0; i < arrayLength(t, t->m->types); ++i) { @@ -1030,21 +1328,26 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, } ++ count; - Type types[count]; - types[0] = Type_object; - unsigned buildOffsets[count]; - buildOffsets[0] = 0; + Field fields[count]; + + new (fields) Field(Type_object, 0, BytesPerWord, 0, TargetBytesPerWord); + unsigned buildOffset = BytesPerWord; - unsigned targetOffsets[count]; - targetOffsets[0] = 0; unsigned targetOffset = TargetBytesPerWord; bool sawArray = false; - unsigned buildSize = BytesPerWord; - unsigned targetSize = TargetBytesPerWord; + Type type; + unsigned buildSize; + unsigned targetSize; for (unsigned j = 1; j < count; ++j) { switch (source[j - 1]) { case Type_object: - types[j] = Type_object; + type = Type_object; + buildSize = BytesPerWord; + targetSize = TargetBytesPerWord; + break; + + case Type_object_nogc: + type = Type_object_nogc; buildSize = BytesPerWord; targetSize = TargetBytesPerWord; break; @@ -1052,39 +1355,39 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, case Type_word: case Type_intptr_t: case Type_uintptr_t: - types[j] = Type_intptr_t; + type = Type_intptr_t; buildSize = BytesPerWord; targetSize = TargetBytesPerWord; break; case Type_int8_t: case Type_uint8_t: - types[j] = Type_int8_t; + type = Type_int8_t; buildSize = targetSize = 1; break; case Type_int16_t: case Type_uint16_t: - types[j] = Type_int16_t; + type = Type_int16_t; buildSize = targetSize = 2; break; case Type_int32_t: case Type_uint32_t: case Type_float: - types[j] = Type_int32_t; + type = Type_int32_t; buildSize = targetSize = 4; break; case Type_int64_t: case Type_uint64_t: case Type_double: - types[j] = Type_int64_t; + type = Type_int64_t; buildSize = targetSize = 8; break; case Type_array: - types[j] = Type_none; + type = Type_none; buildSize = targetSize = 0; break; @@ -1100,16 +1403,14 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, ++ buildOffset; } - buildOffsets[j] = buildOffset; - - buildOffset += buildSize; - while (targetOffset % targetSize) { ++ targetOffset; } - targetOffsets[j] = targetOffset; + new (fields + j) Field + (type, buildOffset, buildSize, targetOffset, targetSize); + buildOffset += buildSize; targetOffset += targetSize; } } @@ -1120,7 +1421,7 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, unsigned targetArrayElementSize; if (sawArray) { fixedFieldCount = count - 2; - arrayElementType = types[count - 1]; + arrayElementType = type; buildArrayElementSize = buildSize; targetArrayElementSize = targetSize; } else { @@ -1141,19 +1442,27 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, targetArrayElementSize, arrayElementType); for (unsigned j = 0; j < fixedFieldCount; ++j) { - expect(t, buildOffsets[j] < map->buildFixedSizeInWords * BytesPerWord); + Field* f = fields + j; - map->targetFixedOffsets()[buildOffsets[j]] = targetOffsets[j]; + expect(t, f->buildOffset + < map->buildFixedSizeInWords * BytesPerWord); - new (map->fixedFields() + j) Field - (types[j], buildOffsets[j], targetOffsets[j]); + map->targetFixedOffsets()[f->buildOffset] = f->targetOffset; + + map->fixedFields()[j] = *f; } - hashMapInsertOrReplace - (t, typeMaps, type(t, static_cast(i)), array, - objectHash, objectEqual); + hashMapInsert + (t, typeMaps, vm::type(t, static_cast(i)), array, + objectHash); } + constants = makeCodeImage + (t, &zone, image, code, codeMap, className, methodName, methodSpec, + typeMaps); + + PROTECT(t, constants); + // these roots will not be used when the bootimage is loaded, so // there's no need to preserve them: setRoot(t, Machine::PoolMap, 0); @@ -1217,9 +1526,10 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, } } - uintptr_t* heap = static_cast + target_uintptr_t* heap = static_cast (t->m->heap->allocate(HeapCapacity)); - uintptr_t* heapMap = static_cast + + target_uintptr_t* heapMap = static_cast (t->m->heap->allocate(heapMapSize(HeapCapacity))); memset(heapMap, 0, heapMapSize(HeapCapacity)); @@ -1275,7 +1585,6 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, heapWalker->dispose(); image->magic = BootImage::Magic; - image->codeBase = reinterpret_cast(code); fprintf(stderr, "class count %d string count %d call count %d\n" "heap size %d code size %d\n", @@ -1301,11 +1610,15 @@ writeBootImage2(Thread* t, FILE* out, BootImage* image, uint8_t* code, ++ offset; } - fwrite(heapMap, pad(heapMapSize(image->heapSize)), 1, out); - fwrite(heap, pad(image->heapSize), 1, out); + fwrite + (heapMap, pad(heapMapSize(image->heapSize), TargetBytesPerWord), 1, out); - fwrite(codeMap, pad(codeMapSize(image->codeSize)), 1, out); - fwrite(code, pad(image->codeSize), 1, out); + fwrite(heap, pad(image->heapSize, TargetBytesPerWord), 1, out); + + fwrite + (codeMap, pad(codeMapSize(image->codeSize), TargetBytesPerWord), 1, out); + + fwrite(code, pad(image->codeSize, TargetBytesPerWord), 1, out); } } diff --git a/src/bootimage.h b/src/bootimage.h index a5ed58bb61..fa3badac08 100644 --- a/src/bootimage.h +++ b/src/bootimage.h @@ -12,16 +12,11 @@ #define BOOTIMAGE_H #include "common.h" +#include "target.h" +#include "machine.h" namespace vm { -const unsigned BootMask = (~static_cast(0)) / BytesPerWord; - -const unsigned BootShift = 32 - log(BytesPerWord); - -const unsigned BootFlatConstant = 1 << BootShift; -const unsigned BootHeapOffset = 1 << (BootShift + 1); - class BootImage { public: class Thunk { @@ -30,14 +25,14 @@ class BootImage { start(0), frameSavedOffset(0), length(0) { } - Thunk(unsigned start, unsigned frameSavedOffset, unsigned length): + Thunk(uint32_t start, uint32_t frameSavedOffset, uint32_t length): start(start), frameSavedOffset(frameSavedOffset), length(length) { } - unsigned start; - unsigned frameSavedOffset; - unsigned length; - }; + uint32_t start; + uint32_t frameSavedOffset; + uint32_t length; + } PACKED; class ThunkCollection { public: @@ -47,63 +42,56 @@ class BootImage { Thunk aioob; Thunk stackOverflow; Thunk table; - }; + } PACKED; - static const unsigned Magic = 0x22377322; + static const uint32_t Magic = 0x22377322; - unsigned magic; + uint32_t magic; - unsigned heapSize; - unsigned codeSize; + uint32_t heapSize; + uint32_t codeSize; - unsigned bootClassCount; - unsigned appClassCount; - unsigned stringCount; - unsigned callCount; + uint32_t bootClassCount; + uint32_t appClassCount; + uint32_t stringCount; + uint32_t callCount; - unsigned bootLoader; - unsigned appLoader; - unsigned types; - unsigned methodTree; - unsigned methodTreeSentinal; - unsigned virtualThunks; + uint32_t bootLoader; + uint32_t appLoader; + uint32_t types; + uint32_t methodTree; + uint32_t methodTreeSentinal; + uint32_t virtualThunks; - uintptr_t codeBase; + uint32_t compileMethodCall; + uint32_t compileVirtualMethodCall; + uint32_t invokeNativeCall; + uint32_t throwArrayIndexOutOfBoundsCall; + uint32_t throwStackOverflowCall; - ThunkCollection thunks; - - unsigned compileMethodCall; - unsigned compileVirtualMethodCall; - unsigned invokeNativeCall; - unsigned throwArrayIndexOutOfBoundsCall; - unsigned throwStackOverflowCall; - -#define THUNK(s) unsigned s##Call; +#define THUNK(s) uint32_t s##Call; #include "thunks.cpp" #undef THUNK + + ThunkCollection thunks; +} PACKED; + +class OffsetResolver { + public: + virtual unsigned fieldOffset(Thread*, object) = 0; }; -inline unsigned -codeMapSize(unsigned codeSize) -{ - return ceiling(codeSize, BitsPerWord) * BytesPerWord; -} +#define NAME(x) Target##x +#define LABEL(x) target_##x +#include "bootimage-template.cpp" +#undef LABEL +#undef NAME -inline unsigned -heapMapSize(unsigned heapSize) -{ - return ceiling(heapSize, BitsPerWord * BytesPerWord) * BytesPerWord; -} - -inline object -bootObject(uintptr_t* heap, unsigned offset) -{ - if (offset) { - return reinterpret_cast(heap + offset - 1); - } else { - return 0; - } -} +#define NAME(x) x +#define LABEL(x) x +#include "bootimage-template.cpp" +#undef LABEL +#undef NAME } // namespace vm diff --git a/src/common.h b/src/common.h index 6a88c6cfa8..118007db47 100644 --- a/src/common.h +++ b/src/common.h @@ -45,6 +45,8 @@ typedef unsigned __int64 uint64_t; # define NO_RETURN __declspec(noreturn) +# define PACKED + # define PLATFORM_WINDOWS # ifdef _M_IX86 @@ -76,6 +78,8 @@ typedef intptr_t intptr_alias_t; # define NO_RETURN __attribute__((noreturn)) +# define PACKED __attribute__((packed)) + # ifdef __MINGW32__ # define PLATFORM_WINDOWS # endif diff --git a/src/compile.cpp b/src/compile.cpp index fa1ffa4778..0dbcb854cc 100644 --- a/src/compile.cpp +++ b/src/compile.cpp @@ -336,7 +336,7 @@ setRoot(Thread* t, Root root, object value); unsigned compiledSize(intptr_t address) { - return reinterpret_cast(address)[-1]; + return reinterpret_cast(address)[-1]; } intptr_t @@ -956,9 +956,10 @@ class BootContext { }; BootContext(Thread* t, object constants, object calls, - DelayedPromise* addresses, Zone* zone): + DelayedPromise* addresses, Zone* zone, OffsetResolver* resolver): protector(t, this), constants(constants), calls(calls), - addresses(addresses), addressSentinal(addresses), zone(zone) + addresses(addresses), addressSentinal(addresses), zone(zone), + resolver(resolver) { } MyProtector protector; @@ -967,6 +968,7 @@ class BootContext { DelayedPromise* addresses; DelayedPromise* addressSentinal; Zone* zone; + OffsetResolver* resolver; }; class Context { @@ -3309,7 +3311,7 @@ compileDirectInvoke(MyThread* t, Frame* frame, object target, bool tailCall, (TargetBytesPerWord, frame->addressOperand(returnAddressPromise), TargetBytesPerWord, c->memory (c->register_(t->arch->thread()), Compiler::AddressType, - difference(&(t->tailAddress), t))); + TargetThreadTailAddress)); c->exit (c->constant @@ -3800,6 +3802,16 @@ intrinsic(MyThread* t, Frame* frame, object target) return false; } +unsigned +targetFieldOffset(Context* context, object field) +{ + if (context->bootContext) { + return context->bootContext->resolver->fieldOffset(context->thread, field); + } else { + return fieldOffset(context->thread, field); + } +} + void compile(MyThread* t, Frame* initialFrame, unsigned ip, int exceptionHandlerStart) @@ -3861,7 +3873,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } if (CheckArrayBounds) { - c->checkBounds(array, ArrayLength, index, aioobThunk(t)); + c->checkBounds(array, TargetArrayLength, index, aioobThunk(t)); } switch (instruction) { @@ -3869,7 +3881,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushObject (c->load (TargetBytesPerWord, TargetBytesPerWord, c->memory - (array, Compiler::ObjectType, ArrayBody, index, + (array, Compiler::ObjectType, TargetArrayBody, index, TargetBytesPerWord), TargetBytesPerWord)); break; @@ -3878,7 +3890,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushInt (c->load (4, 4, c->memory - (array, Compiler::FloatType, ArrayBody, index, 4), + (array, Compiler::FloatType, TargetArrayBody, index, 4), TargetBytesPerWord)); break; @@ -3886,7 +3898,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushInt (c->load (4, 4, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 4), + (array, Compiler::IntegerType, TargetArrayBody, index, 4), TargetBytesPerWord)); break; @@ -3894,7 +3906,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushInt (c->load (1, 1, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 1), + (array, Compiler::IntegerType, TargetArrayBody, index, 1), TargetBytesPerWord)); break; @@ -3902,7 +3914,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushInt (c->loadz (2, 2, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 2), + (array, Compiler::IntegerType, TargetArrayBody, index, 2), TargetBytesPerWord)); break; @@ -3910,21 +3922,21 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushLong (c->load (8, 8, c->memory - (array, Compiler::FloatType, ArrayBody, index, 8), 8)); + (array, Compiler::FloatType, TargetArrayBody, index, 8), 8)); break; case laload: frame->pushLong (c->load (8, 8, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 8), 8)); + (array, Compiler::IntegerType, TargetArrayBody, index, 8), 8)); break; case saload: frame->pushInt (c->load (2, 2, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 2), + (array, Compiler::IntegerType, TargetArrayBody, index, 2), TargetBytesPerWord)); break; } @@ -3956,7 +3968,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, } if (CheckArrayBounds) { - c->checkBounds(array, ArrayLength, index, aioobThunk(t)); + c->checkBounds(array, TargetArrayLength, index, aioobThunk(t)); } switch (instruction) { @@ -3969,7 +3981,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, Compiler::VoidType, 4, c->register_(t->arch->thread()), array, c->add - (4, c->constant(ArrayBody, Compiler::IntegerType), + (4, c->constant(TargetArrayBody, Compiler::IntegerType), c->shl (4, c->constant(log(TargetBytesPerWord), Compiler::IntegerType), index)), @@ -3979,38 +3991,38 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, case fastore: c->store (TargetBytesPerWord, value, 4, c->memory - (array, Compiler::FloatType, ArrayBody, index, 4)); + (array, Compiler::FloatType, TargetArrayBody, index, 4)); break; case iastore: c->store (TargetBytesPerWord, value, 4, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 4)); + (array, Compiler::IntegerType, TargetArrayBody, index, 4)); break; case bastore: c->store (TargetBytesPerWord, value, 1, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 1)); + (array, Compiler::IntegerType, TargetArrayBody, index, 1)); break; case castore: case sastore: c->store (TargetBytesPerWord, value, 2, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 2)); + (array, Compiler::IntegerType, TargetArrayBody, index, 2)); break; case dastore: c->store (8, value, 8, c->memory - (array, Compiler::FloatType, ArrayBody, index, 8)); + (array, Compiler::FloatType, TargetArrayBody, index, 8)); break; case lastore: c->store (8, value, 8, c->memory - (array, Compiler::IntegerType, ArrayBody, index, 8)); + (array, Compiler::IntegerType, TargetArrayBody, index, 8)); break; } } break; @@ -4082,7 +4094,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (TargetBytesPerWord, TargetBytesPerWord, c->memory - (frame->popObject(), Compiler::IntegerType, ArrayLength, 0, 1), + (frame->popObject(), Compiler::IntegerType, + TargetArrayLength, 0, 1), TargetBytesPerWord)); } break; @@ -4425,54 +4438,56 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, frame->pushInt (c->load (1, 1, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - TargetBytesPerWord)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1), TargetBytesPerWord)); break; case CharField: frame->pushInt (c->loadz (2, 2, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - TargetBytesPerWord)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1), TargetBytesPerWord)); break; case ShortField: frame->pushInt (c->load (2, 2, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - TargetBytesPerWord)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1), TargetBytesPerWord)); break; case FloatField: frame->pushInt (c->load (4, 4, c->memory - (table, Compiler::FloatType, fieldOffset(t, field), 0, 1), - TargetBytesPerWord)); + (table, Compiler::FloatType, targetFieldOffset + (context, field), 0, 1), TargetBytesPerWord)); break; case IntField: frame->pushInt (c->load (4, 4, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), - TargetBytesPerWord)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1), TargetBytesPerWord)); break; case DoubleField: frame->pushLong (c->load (8, 8, c->memory - (table, Compiler::FloatType, fieldOffset(t, field), 0, 1), 8)); + (table, Compiler::FloatType, targetFieldOffset + (context, field), 0, 1), 8)); break; case LongField: frame->pushLong (c->load (8, 8, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1), 8)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1), 8)); break; case ObjectField: @@ -4480,8 +4495,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->load (TargetBytesPerWord, TargetBytesPerWord, c->memory - (table, Compiler::ObjectType, fieldOffset(t, field), 0, 1), - TargetBytesPerWord)); + (table, Compiler::ObjectType, targetFieldOffset + (context, field), 0, 1), TargetBytesPerWord)); break; default: @@ -4974,8 +4989,8 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, if (LIKELY(methodVirtual(t, target))) { unsigned parameterFootprint = methodParameterFootprint(t, target); - unsigned offset = ClassVtable - + (methodOffset(t, target) * BytesPerWord); + unsigned offset = TargetClassVtable + + (methodOffset(t, target) * TargetBytesPerWord); Compiler::Operand* instance = c->peek(1, parameterFootprint - 1); @@ -4985,7 +5000,7 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, (c->memory (c->and_ (TargetBytesPerWord, c->constant - (PointerMask, Compiler::IntegerType), + (TargetPointerMask, Compiler::IntegerType), c->memory(instance, Compiler::ObjectType, 0, 0, 1)), Compiler::ObjectType, offset, 0, 1), tailCall ? Compiler::TailJump : 0, @@ -5635,38 +5650,44 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, case BooleanField: c->store (TargetBytesPerWord, value, 1, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1)); break; case CharField: case ShortField: c->store (TargetBytesPerWord, value, 2, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1)); break; case FloatField: c->store (TargetBytesPerWord, value, 4, c->memory - (table, Compiler::FloatType, fieldOffset(t, field), 0, 1)); + (table, Compiler::FloatType, targetFieldOffset + (context, field), 0, 1)); break; case IntField: c->store (TargetBytesPerWord, value, 4, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1)); break; case DoubleField: c->store (8, value, 8, c->memory - (table, Compiler::FloatType, fieldOffset(t, field), 0, 1)); + (table, Compiler::FloatType, targetFieldOffset + (context, field), 0, 1)); break; case LongField: c->store (8, value, 8, c->memory - (table, Compiler::IntegerType, fieldOffset(t, field), 0, 1)); + (table, Compiler::IntegerType, targetFieldOffset + (context, field), 0, 1)); break; case ObjectField: @@ -5679,14 +5700,16 @@ compile(MyThread* t, Frame* initialFrame, unsigned ip, 0, Compiler::VoidType, 4, c->register_(t->arch->thread()), table, - c->constant(fieldOffset(t, field), Compiler::IntegerType), + c->constant(targetFieldOffset(context, field), + Compiler::IntegerType), value); } else { c->call (c->constant(getThunk(t, setThunk), Compiler::AddressType), 0, 0, 0, Compiler::VoidType, 4, c->register_(t->arch->thread()), table, - c->constant(fieldOffset(t, field), Compiler::IntegerType), + c->constant(targetFieldOffset(context, field), + Compiler::IntegerType), value); } break; @@ -6797,7 +6820,7 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) // of cycles if another thread compiles the same method in parallel, // which might be mitigated by fine-grained, per-method locking): c->compile(context->leaf ? 0 : stackOverflowThunk(t), - difference(&(t->stackLimit), t)); + TargetThreadStackLimit); // we must acquire the class lock here at the latest @@ -6806,7 +6829,8 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) unsigned total = pad(codeSize) + pad(c->poolSize()) + TargetBytesPerWord; - uintptr_t* code = static_cast(allocator->allocate(total)); + target_uintptr_t* code = static_cast + (allocator->allocate(total)); code[0] = codeSize; uint8_t* start = reinterpret_cast(code + 1); @@ -6817,8 +6841,7 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) if (context->objectPool) { object pool = allocate3 (t, allocator, Machine::ImmortalAllocation, - FixedSizeOfArray - + ((context->objectPoolCount + 1) * TargetBytesPerWord), + FixedSizeOfArray + ((context->objectPoolCount + 1) * BytesPerWord), true); initArray(t, pool, context->objectPoolCount + 1); @@ -6829,7 +6852,7 @@ finish(MyThread* t, FixedAllocator* allocator, Context* context) unsigned i = 1; for (PoolElement* p = context->objectPool; p; p = p->next) { - unsigned offset = ArrayBody + ((i++) * TargetBytesPerWord); + unsigned offset = ArrayBody + ((i++) * BytesPerWord); p->address = reinterpret_cast(pool) + offset; @@ -8318,13 +8341,30 @@ class MyProcessor: public Processor { t->init(); if (false) { - fprintf(stderr, "%d\n", difference(&(t->stack), t)); - fprintf(stderr, "%d\n", difference(&(t->scratch), t)); - fprintf(stderr, "%d\n", difference(&(t->continuation), t)); - fprintf(stderr, "%d\n", difference(&(t->exception), t)); - fprintf(stderr, "%d\n", difference(&(t->exceptionStackAdjustment), t)); - fprintf(stderr, "%d\n", difference(&(t->exceptionOffset), t)); - fprintf(stderr, "%d\n", difference(&(t->exceptionHandler), t)); + fprintf(stderr, "stack %d\n", + difference(&(t->stack), t)); + fprintf(stderr, "scratch %d\n", + difference(&(t->scratch), t)); + fprintf(stderr, "continuation %d\n", + difference(&(t->continuation), t)); + fprintf(stderr, "exception %d\n", + difference(&(t->exception), t)); + fprintf(stderr, "exceptionStackAdjustment %d\n", + difference(&(t->exceptionStackAdjustment), t)); + fprintf(stderr, "exceptionOffset %d\n", + difference(&(t->exceptionOffset), t)); + fprintf(stderr, "exceptionHandler %d\n", + difference(&(t->exceptionHandler), t)); + fprintf(stderr, "tailAddress %d\n", + difference(&(t->tailAddress), t)); + fprintf(stderr, "stackLimit %d\n", + difference(&(t->stackLimit), t)); + fprintf(stderr, "ip %d\n", + difference(&(t->ip), t)); + fprintf(stderr, "virtualCallTarget %d\n", + difference(&(t->virtualCallTarget), t)); + fprintf(stderr, "virtualCallIndex %d\n", + difference(&(t->virtualCallIndex), t)); exit(0); } @@ -8671,10 +8711,10 @@ class MyProcessor: public Processor { virtual void compileMethod(Thread* vmt, Zone* zone, object* constants, object* calls, DelayedPromise** addresses, - object method) + object method, OffsetResolver* resolver) { MyThread* t = static_cast(vmt); - BootContext bootContext(t, *constants, *calls, *addresses, zone); + BootContext bootContext(t, *constants, *calls, *addresses, zone, resolver); compile(t, &codeAllocator, &bootContext, method); @@ -8689,6 +8729,17 @@ class MyProcessor: public Processor { bootImage->virtualThunks = w->visitRoot(root(t, VirtualThunks)); } + virtual void normalizeVirtualThunks(Thread* t) { + for (unsigned i = 0; i < wordArrayLength(t, root(t, VirtualThunks)); + i += 2) + { + if (wordArrayBody(t, root(t, VirtualThunks), i)) { + wordArrayBody(t, root(t, VirtualThunks), i) + -= reinterpret_cast(codeAllocator.base); + } + } + } + virtual unsigned* makeCallTable(Thread* t, HeapWalker* w) { bootImage->codeSize = codeAllocator.offset; bootImage->callCount = callTableSize; @@ -8704,7 +8755,7 @@ class MyProcessor: public Processor { table[index++] = callNodeAddress(t, p) - reinterpret_cast(codeAllocator.base); table[index++] = w->map()->find(callNodeTarget(t, p)) - | (static_cast(callNodeFlags(t, p)) << BootShift); + | (static_cast(callNodeFlags(t, p)) << TargetBootShift); } } @@ -8989,7 +9040,8 @@ insertCallNode(MyThread* t, object node) } object -makeClassMap(Thread* t, unsigned* table, unsigned count, uintptr_t* heap) +makeClassMap(Thread* t, unsigned* table, unsigned count, + uintptr_t* heap) { object array = makeArray(t, nextPowerOfTwo(count)); object map = makeHashMap(t, 0, array); @@ -9068,6 +9120,7 @@ fixupHeap(MyThread* t UNUSED, uintptr_t* map, unsigned size, uintptr_t* heap) for (unsigned bit = 0; bit < BitsPerWord; ++bit) { if (w & (static_cast(1) << bit)) { unsigned index = indexOf(word, bit); + uintptr_t* p = heap + index; assert(t, *p); @@ -9103,6 +9156,10 @@ fixupCode(Thread* t, uintptr_t* map, unsigned size, uint8_t* code, if (oldValue & BootHeapOffset) { newValue = reinterpret_cast (heap + (oldValue & BootMask) - 1); + + // fprintf(stderr, "constant marked %d %d\n", + // index, static_cast(oldValue)); + } else { newValue = reinterpret_cast (code + (oldValue & BootMask)); @@ -9130,12 +9187,11 @@ fixupMethods(Thread* t, object map, BootImage* image, uint8_t* code) for (unsigned i = 0; i < arrayLength(t, classMethodTable(t, c)); ++i) { object method = arrayBody(t, classMethodTable(t, c), i); if (methodCode(t, method)) { - assert(t, (methodCompiled(t, method) - image->codeBase) - <= image->codeSize); + assert(t, methodCompiled(t, method) + <= static_cast(image->codeSize)); codeCompiled(t, methodCode(t, method)) - = (methodCompiled(t, method) - image->codeBase) - + reinterpret_cast(code); + = methodCompiled(t, method) + reinterpret_cast(code); if (DebugCompile) { logCompile @@ -9203,13 +9259,13 @@ fixupThunks(MyThread* t, BootImage* image, uint8_t* code) } void -fixupVirtualThunks(MyThread* t, BootImage* image, uint8_t* code) +fixupVirtualThunks(MyThread* t, uint8_t* code) { for (unsigned i = 0; i < wordArrayLength(t, root(t, VirtualThunks)); i += 2) { if (wordArrayBody(t, root(t, VirtualThunks), i)) { wordArrayBody(t, root(t, VirtualThunks), i) - = (wordArrayBody(t, root(t, VirtualThunks), i) - image->codeBase) + = wordArrayBody(t, root(t, VirtualThunks), i) + reinterpret_cast(code); } } @@ -9231,16 +9287,16 @@ boot(MyThread* t, BootImage* image) (heapMapSize(image->heapSize), BytesPerWord); uintptr_t* heap = heapMap + heapMapSizeInWords; -// fprintf(stderr, "heap from %p to %p\n", -// heap, heap + ceiling(image->heapSize, BytesPerWord)); + // fprintf(stderr, "heap from %p to %p\n", + // heap, heap + ceiling(image->heapSize, BytesPerWord)); uintptr_t* codeMap = heap + ceiling(image->heapSize, BytesPerWord); unsigned codeMapSizeInWords = ceiling (codeMapSize(image->codeSize), BytesPerWord); uint8_t* code = reinterpret_cast(codeMap + codeMapSizeInWords); -// fprintf(stderr, "code from %p to %p\n", -// code, code + image->codeSize); + // fprintf(stderr, "code from %p to %p\n", + // code, code + image->codeSize); fixupHeap(t, heapMap, heapMapSizeInWords, heap); @@ -9293,7 +9349,7 @@ boot(MyThread* t, BootImage* image) fixupThunks(t, image, code); - fixupVirtualThunks(t, image, code); + fixupVirtualThunks(t, code); fixupMethods (t, classLoaderMap(t, root(t, Machine::BootLoader)), image, code); @@ -9337,7 +9393,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) { Assembler* a = defaultContext.context.assembler; - a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); + a->saveFrame(TargetThreadStack, TargetThreadIp); p->thunks.default_.frameSavedOffset = a->length(); @@ -9369,19 +9425,19 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) TargetBytesPerWord, RegisterOperand, &class_); Assembler::Memory virtualCallTargetDst - (t->arch->thread(), difference(&(t->virtualCallTarget), t)); + (t->arch->thread(), TargetThreadVirtualCallTarget); a->apply(Move, TargetBytesPerWord, RegisterOperand, &class_, TargetBytesPerWord, MemoryOperand, &virtualCallTargetDst); Assembler::Register index(t->arch->virtualCallIndex()); Assembler::Memory virtualCallIndex - (t->arch->thread(), difference(&(t->virtualCallIndex), t)); + (t->arch->thread(), TargetThreadVirtualCallIndex); a->apply(Move, TargetBytesPerWord, RegisterOperand, &index, TargetBytesPerWord, MemoryOperand, &virtualCallIndex); - a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); + a->saveFrame(TargetThreadStack, TargetThreadIp); p->thunks.defaultVirtual.frameSavedOffset = a->length(); @@ -9403,7 +9459,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) { Assembler* a = nativeContext.context.assembler; - a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); + a->saveFrame(TargetThreadStack, TargetThreadIp); p->thunks.native.frameSavedOffset = a->length(); @@ -9414,7 +9470,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) a->apply(LongCall, TargetBytesPerWord, ConstantOperand, &proc); a->popFrameAndUpdateStackAndReturn - (t->arch->alignFrameSize(1), difference(&(t->stack), t)); + (t->arch->alignFrameSize(1), TargetThreadStack); p->thunks.native.length = a->endBlock(false)->resolve(0, 0); } @@ -9423,7 +9479,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) { Assembler* a = aioobContext.context.assembler; - a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); + a->saveFrame(TargetThreadStack, TargetThreadIp); p->thunks.aioob.frameSavedOffset = a->length(); @@ -9440,7 +9496,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) { Assembler* a = stackOverflowContext.context.assembler; - a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); + a->saveFrame(TargetThreadStack, TargetThreadIp); p->thunks.stackOverflow.frameSavedOffset = a->length(); @@ -9457,7 +9513,7 @@ compileThunks(MyThread* t, Allocator* allocator, MyProcessor* p) { Assembler* a = tableContext.context.assembler; - a->saveFrame(difference(&(t->stack), t), difference(&(t->ip), t)); + a->saveFrame(TargetThreadStack, TargetThreadIp); p->thunks.table.frameSavedOffset = a->length(); diff --git a/src/compiler.cpp b/src/compiler.cpp index 157bb82d61..5221485cac 100644 --- a/src/compiler.cpp +++ b/src/compiler.cpp @@ -469,7 +469,7 @@ class PoolPromise: public Promise { virtual int64_t value() { if (resolved()) { return reinterpret_cast - (c->machineCode + pad(c->machineCodeSize) + (c->machineCode + pad(c->machineCodeSize, TargetBytesPerWord) + (key * TargetBytesPerWord)); } @@ -6949,15 +6949,15 @@ class MyCompiler: public Compiler { int i = 0; for (ConstantPoolNode* n = c.firstConstant; n; n = n->next) { - intptr_t* target = reinterpret_cast - (c.machineCode + pad(c.machineCodeSize) + i); + target_intptr_t* target = reinterpret_cast + (c.machineCode + pad(c.machineCodeSize, TargetBytesPerWord) + i); if (n->promise->resolved()) { *target = n->promise->value(); } else { class Listener: public Promise::Listener { public: - Listener(intptr_t* target): target(target){ } + Listener(target_intptr_t* target): target(target){ } virtual bool resolve(int64_t value, void** location) { *target = value; @@ -6965,7 +6965,7 @@ class MyCompiler: public Compiler { return true; } - intptr_t* target; + target_intptr_t* target; }; new (n->promise->listen(sizeof(Listener))) Listener(target); } diff --git a/src/heap.cpp b/src/heap.cpp index 8208cd7e73..c8f30c0369 100644 --- a/src/heap.cpp +++ b/src/heap.cpp @@ -529,7 +529,7 @@ class Fixie { } static unsigned maskSize(unsigned size, bool hasMask) { - return hasMask * ceiling(size, BytesPerWord) * BytesPerWord; + return hasMask * ceiling(size, BitsPerWord) * BytesPerWord; } static unsigned totalSize(unsigned size, bool hasMask) { diff --git a/src/interpret.cpp b/src/interpret.cpp index 255977c25d..45a3105753 100644 --- a/src/interpret.cpp +++ b/src/interpret.cpp @@ -3067,7 +3067,7 @@ class MyProcessor: public Processor { } virtual void compileMethod(vm::Thread*, Zone*, object*, object*, - DelayedPromise**, object) + DelayedPromise**, object, OffsetResolver*) { abort(s); } @@ -3076,6 +3076,10 @@ class MyProcessor: public Processor { abort(s); } + virtual void normalizeVirtualThunks(vm::Thread*) { + abort(s); + } + virtual unsigned* makeCallTable(vm::Thread*, HeapWalker*) { abort(s); } diff --git a/src/processor.h b/src/processor.h index f790491439..4163590846 100644 --- a/src/processor.h +++ b/src/processor.h @@ -122,11 +122,15 @@ class Processor { virtual void compileMethod(Thread* t, Zone* zone, object* constants, object* calls, - DelayedPromise** addresses, object method) = 0; + DelayedPromise** addresses, object method, + OffsetResolver* resolver) = 0; virtual void visitRoots(Thread* t, HeapWalker* w) = 0; + virtual void + normalizeVirtualThunks(Thread* t) = 0; + virtual unsigned* makeCallTable(Thread* t, HeapWalker* w) = 0; diff --git a/src/target.h b/src/target.h index 91244b7ed6..ec309c8cd0 100644 --- a/src/target.h +++ b/src/target.h @@ -42,25 +42,62 @@ namespace vm { #ifdef TARGET_BYTES_PER_WORD # if (TARGET_BYTES_PER_WORD == 8) # define TARGET_VW(v) TARGET_V8(v) + typedef uint64_t target_uintptr_t; typedef int64_t target_intptr_t; + const unsigned TargetBytesPerWord = 8; + +const unsigned TargetThreadTailAddress = 2272; +const unsigned TargetThreadStackLimit = 2336; +const unsigned TargetThreadStack = 2224; +const unsigned TargetThreadIp = 2216; +const unsigned TargetThreadVirtualCallTarget = 2280; +const unsigned TargetThreadVirtualCallIndex = 2288; + +const unsigned TargetClassFixedSize = 12; +const unsigned TargetClassArrayElementSize = 14; +const unsigned TargetClassVtable = 128; + +const unsigned TargetFieldOffset = 12; + # elif (TARGET_BYTES_PER_WORD == 4) # define TARGET_VW(v) TARGET_V4(v) + typedef uint32_t target_uintptr_t; typedef int32_t target_intptr_t; + const unsigned TargetBytesPerWord = 4; + +const unsigned TargetThreadTailAddress = 2172; +const unsigned TargetThreadStackLimit = 2204; +const unsigned TargetThreadStack = 2148; +const unsigned TargetThreadIp = 2144; +const unsigned TargetThreadVirtualCallTarget = 2176; +const unsigned TargetThreadVirtualCallIndex = 2180; + +const unsigned TargetClassFixedSize = 8; +const unsigned TargetClassArrayElementSize = 10; +const unsigned TargetClassVtable = 68; + +const unsigned TargetFieldOffset = 8; + # else # error # endif #else -typedef uintptr_t target_uintptr_t; -typedef intptr_t target_intptr_t; -const unsigned TargetBytesPerWord = BytesPerWord; +# error #endif const unsigned TargetBitsPerWord = TargetBytesPerWord * 8; +const uintptr_t TargetPointerMask += ((~static_cast(0)) / TargetBytesPerWord) + * TargetBytesPerWord; + +const unsigned TargetArrayLength = TargetBytesPerWord; +const unsigned TargetArrayBody = TargetBytesPerWord * 2; + } // namespace vm #endif//TARGET_H diff --git a/src/type-generator.cpp b/src/type-generator.cpp index eb0141f215..52eef613b6 100644 --- a/src/type-generator.cpp +++ b/src/type-generator.cpp @@ -2116,6 +2116,9 @@ writeMap(Output* out, Object* type) case Object::Scalar: { out->write("Type_"); out->write(memberTypeEnumName(m)); + if (memberNoGC(m)) { + out->write("_nogc"); + } } break; case Object::Array: { diff --git a/src/vector.h b/src/vector.h index a83aed9a7c..6714573b0d 100644 --- a/src/vector.h +++ b/src/vector.h @@ -12,6 +12,7 @@ #define VECTOR_H #include "system.h" +#include "target.h" namespace vm { @@ -103,6 +104,10 @@ class Vector { append(&v, 4); } + void appendTargetAddress(target_uintptr_t v) { + append(&v, TargetBytesPerWord); + } + void appendAddress(uintptr_t v) { append(&v, BytesPerWord); } diff --git a/src/x86.cpp b/src/x86.cpp index 5a90ede640..a1c1a833b5 100644 --- a/src/x86.cpp +++ b/src/x86.cpp @@ -74,13 +74,13 @@ const unsigned StackAlignmentInBytes = 16; const unsigned StackAlignmentInWords = StackAlignmentInBytes / TargetBytesPerWord; bool -isInt8(intptr_t v) +isInt8(target_intptr_t v) { return v == static_cast(v); } bool -isInt32(intptr_t v) +isInt32(target_intptr_t v) { return v == static_cast(v); } @@ -958,11 +958,11 @@ moveCR2(Context* c, UNUSED unsigned aSize, Assembler::Constant* a, maybeRex(c, TargetBytesPerWord, b); opcode(c, 0xb8 + regCode(b)); if (a->value->resolved()) { - c->code.appendAddress(a->value->value()); + c->code.appendTargetAddress(a->value->value()); } else { appendImmediateTask (c, a->value, offset(c), TargetBytesPerWord, promiseOffset); - c->code.appendAddress(static_cast(0)); + c->code.appendTargetAddress(static_cast(0)); } } } @@ -3446,7 +3446,7 @@ class MyAssembler: public Assembler { RUNTIME_ARRAY_BODY(arguments)[i].size, RUNTIME_ARRAY_BODY(arguments)[i].type, RUNTIME_ARRAY_BODY(arguments)[i].operand, - pad(RUNTIME_ARRAY_BODY(arguments)[i].size), + pad(RUNTIME_ARRAY_BODY(arguments)[i].size, TargetBytesPerWord), RegisterOperand, &dst); } else { @@ -3455,7 +3455,7 @@ class MyAssembler: public Assembler { RUNTIME_ARRAY_BODY(arguments)[i].size, RUNTIME_ARRAY_BODY(arguments)[i].type, RUNTIME_ARRAY_BODY(arguments)[i].operand, - pad(RUNTIME_ARRAY_BODY(arguments)[i].size), + pad(RUNTIME_ARRAY_BODY(arguments)[i].size, TargetBytesPerWord), MemoryOperand, &dst); offset += ceiling From e3fc8d3dd282bdaab059753332cd4646bbe1f1ed Mon Sep 17 00:00:00 2001 From: Joel Dice Date: Thu, 1 Sep 2011 10:51:56 -0600 Subject: [PATCH 5/5] enable cross-architecture ARM bootimage builds --- src/arm.cpp | 177 ++++++++++++++++++++++++++++------------------------ 1 file changed, 94 insertions(+), 83 deletions(-) diff --git a/src/arm.cpp b/src/arm.cpp index f7a50b815a..0b1b8cd335 100644 --- a/src/arm.cpp +++ b/src/arm.cpp @@ -155,11 +155,11 @@ inline int unha16(int32_t high, int32_t low) { return ((high - ((low & 0x8000) ? 1 : 0)) << 16) | low; } -inline bool isInt8(intptr_t v) { return v == static_cast(v); } -inline bool isInt16(intptr_t v) { return v == static_cast(v); } -inline bool isInt24(intptr_t v) { return v == (v & 0xffffff); } -inline bool isInt32(intptr_t v) { return v == static_cast(v); } -inline int carry16(intptr_t v) { return static_cast(v) < 0 ? 1 : 0; } +inline bool isInt8(target_intptr_t v) { return v == static_cast(v); } +inline bool isInt16(target_intptr_t v) { return v == static_cast(v); } +inline bool isInt24(target_intptr_t v) { return v == (v & 0xffffff); } +inline bool isInt32(target_intptr_t v) { return v == static_cast(v); } +inline int carry16(target_intptr_t v) { return static_cast(v) < 0 ? 1 : 0; } inline bool isOfWidth(long long i, int size) { return static_cast(i) >> size == 0; } inline bool isOfWidth(int i, int size) { return static_cast(i) >> size == 0; } @@ -167,7 +167,8 @@ inline bool isOfWidth(int i, int size) { return static_cast(i) >> size const unsigned FrameHeaderSize = 1; const unsigned StackAlignmentInBytes = 8; -const unsigned StackAlignmentInWords = StackAlignmentInBytes / BytesPerWord; +const unsigned StackAlignmentInWords += StackAlignmentInBytes / TargetBytesPerWord; const int ThreadRegister = 8; const int StackRegister = 13; @@ -331,7 +332,8 @@ class Offset: public Promise { assert(c, resolved()); unsigned o = offset - block->offset; - return block->start + padding(block, forTrace ? o - BytesPerWord : o) + o; + return block->start + padding + (block, forTrace ? o - TargetBytesPerWord : o) + o; } Context* c; @@ -496,7 +498,7 @@ void shiftLeftR(Context* con, unsigned size, Assembler::Register* a, Assembler:: void shiftLeftC(Context* con, unsigned size UNUSED, Assembler::Constant* a, Assembler::Register* b, Assembler::Register* t) { - assert(con, size == BytesPerWord); + assert(con, size == TargetBytesPerWord); emit(con, lsli(t->low, b->low, getValue(a))); } @@ -519,7 +521,7 @@ void shiftRightR(Context* con, unsigned size, Assembler::Register* a, Assembler: void shiftRightC(Context* con, unsigned size UNUSED, Assembler::Constant* a, Assembler::Register* b, Assembler::Register* t) { - assert(con, size == BytesPerWord); + assert(con, size == TargetBytesPerWord); emit(con, asri(t->low, b->low, getValue(a))); } @@ -541,7 +543,7 @@ void unsignedShiftRightR(Context* con, unsigned size, Assembler::Register* a, As void unsignedShiftRightC(Context* con, unsigned size UNUSED, Assembler::Constant* a, Assembler::Register* b, Assembler::Register* t) { - assert(con, size == BytesPerWord); + assert(con, size == TargetBytesPerWord); emit(con, lsri(t->low, b->low, getValue(a))); } @@ -556,7 +558,7 @@ class ConstantPoolEntry: public Promise { virtual int64_t value() { assert(c, resolved()); - return reinterpret_cast(address); + return reinterpret_cast(address); } virtual bool resolved() { @@ -573,7 +575,8 @@ class ConstantPoolEntry: public Promise { class ConstantPoolListener: public Promise::Listener { public: - ConstantPoolListener(System* s, uintptr_t* address, uint8_t* returnAddress): + ConstantPoolListener(System* s, target_uintptr_t* address, + uint8_t* returnAddress): s(s), address(address), returnAddress(returnAddress) @@ -588,7 +591,7 @@ class ConstantPoolListener: public Promise::Listener { } System* s; - uintptr_t* address; + target_uintptr_t* address; uint8_t* returnAddress; }; @@ -677,10 +680,10 @@ padding(MyBlock* b, unsigned offset) for (PoolEvent* e = b->poolEventHead; e; e = e->next) { if (e->offset <= offset) { if (needJump(b)) { - total += BytesPerWord; + total += TargetBytesPerWord; } for (PoolOffset* o = e->poolOffsetHead; o; o = o->next) { - total += BytesPerWord; + total += TargetBytesPerWord; } } else { break; @@ -708,7 +711,7 @@ resolve(MyBlock* b) if (b->next == 0 or b->next->poolEventHead) { append = true; } else { - int32_t v = (b->start + b->size + b->next->size + BytesPerWord - 8) + int32_t v = (b->start + b->size + b->next->size + TargetBytesPerWord - 8) - (c->poolOffsetHead->offset + c->poolOffsetHead->block->start); append = (v != (v & PoolOffsetMask)); @@ -751,7 +754,7 @@ resolve(MyBlock* b) void jumpR(Context* c, unsigned size UNUSED, Assembler::Register* target) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); emit(c, bx(target->low)); } @@ -763,8 +766,8 @@ void swapRR(Context* c, unsigned aSize, Assembler::Register* a, unsigned bSize, Assembler::Register* b) { - assert(c, aSize == BytesPerWord); - assert(c, bSize == BytesPerWord); + assert(c, aSize == TargetBytesPerWord); + assert(c, bSize == TargetBytesPerWord); Assembler::Register tmp(c->client->acquireTemporary()); moveRR(c, aSize, a, bSize, &tmp); @@ -876,7 +879,7 @@ void addC(Context* c, unsigned size, Assembler::Constant* a, Assembler::Register* b, Assembler::Register* dst) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); int32_t v = a->value->value(); if (v) { @@ -897,7 +900,7 @@ void subC(Context* c, unsigned size, Assembler::Constant* a, Assembler::Register* b, Assembler::Register* dst) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); int32_t v = a->value->value(); if (v) { @@ -958,7 +961,7 @@ normalize(Context* c, int offset, int index, unsigned scale, ResolvedPromise scalePromise(log(scale)); Assembler::Constant scaleConstant(&scalePromise); - shiftLeftC(c, BytesPerWord, &scaleConstant, + shiftLeftC(c, TargetBytesPerWord, &scaleConstant, &unscaledIndex, &normalizedIndex); scaled = normalizedIndex.low; @@ -973,8 +976,8 @@ normalize(Context* c, int offset, int index, unsigned scale, Assembler::Constant offsetConstant(&offsetPromise); Assembler::Register tmp(c->client->acquireTemporary()); - moveCR(c, BytesPerWord, &offsetConstant, BytesPerWord, &tmp); - addR(c, BytesPerWord, &tmp, &untranslatedIndex, &normalizedIndex); + moveCR(c, TargetBytesPerWord, &offsetConstant, TargetBytesPerWord, &tmp); + addR(c, TargetBytesPerWord, &tmp, &untranslatedIndex, &normalizedIndex); c->client->releaseTemporary(tmp.low); } @@ -1046,7 +1049,7 @@ store(Context* c, unsigned size, Assembler::Register* src, Assembler::Register tmp(c->client->acquireTemporary()); ResolvedPromise offsetPromise(offset); Assembler::Constant offsetConstant(&offsetPromise); - moveCR(c, BytesPerWord, &offsetConstant, BytesPerWord, &tmp); + moveCR(c, TargetBytesPerWord, &offsetConstant, TargetBytesPerWord, &tmp); store(c, size, src, base, 0, tmp.low, 1, false); @@ -1067,8 +1070,8 @@ void moveAndUpdateRM(Context* c, unsigned srcSize UNUSED, Assembler::Register* src, unsigned dstSize UNUSED, Assembler::Memory* dst) { - assert(c, srcSize == BytesPerWord); - assert(c, dstSize == BytesPerWord); + assert(c, srcSize == TargetBytesPerWord); + assert(c, dstSize == TargetBytesPerWord); if (dst->index == NoRegister) { emit(c, stri(src->low, dst->base, dst->offset, dst->offset ? 1 : 0)); @@ -1168,7 +1171,7 @@ load(Context* c, unsigned srcSize, int base, int offset, int index, Assembler::Register tmp(c->client->acquireTemporary()); ResolvedPromise offsetPromise(offset); Assembler::Constant offsetConstant(&offsetPromise); - moveCR(c, BytesPerWord, &offsetConstant, BytesPerWord, &tmp); + moveCR(c, TargetBytesPerWord, &offsetConstant, TargetBytesPerWord, &tmp); load(c, srcSize, base, 0, tmp.low, 1, dstSize, dst, false, signExtend); @@ -1457,7 +1460,7 @@ branchRR(Context* c, TernaryOperation op, unsigned size, Assembler::Register* a, Assembler::Register* b, Assembler::Constant* target) { - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { Assembler::Register ah(a->high); Assembler::Register bh(b->high); @@ -1474,13 +1477,13 @@ branchCR(Context* c, TernaryOperation op, unsigned size, Assembler::Constant* a, Assembler::Register* b, Assembler::Constant* target) { - if (size > BytesPerWord) { + if (size > TargetBytesPerWord) { int64_t v = a->value->value(); - ResolvedPromise low(v & ~static_cast(0)); + ResolvedPromise low(v & ~static_cast(0)); Assembler::Constant al(&low); - ResolvedPromise high((v >> 32) & ~static_cast(0)); + ResolvedPromise high((v >> 32) & ~static_cast(0)); Assembler::Constant ah(&high); Assembler::Register bh(b->high); @@ -1498,7 +1501,7 @@ branchRM(Context* c, TernaryOperation op, unsigned size, Assembler::Register* a, Assembler::Memory* b, Assembler::Constant* target) { - assert(c, size <= BytesPerWord); + assert(c, size <= TargetBytesPerWord); compareRM(c, size, a, size, b); branch(c, op, target); @@ -1509,7 +1512,7 @@ branchCM(Context* c, TernaryOperation op, unsigned size, Assembler::Constant* a, Assembler::Memory* b, Assembler::Constant* target) { - assert(c, size <= BytesPerWord); + assert(c, size <= TargetBytesPerWord); compareCM(c, size, a, size, b); branch(c, op, target); @@ -1565,14 +1568,14 @@ negateRR(Context* c, unsigned srcSize, Assembler::Register* src, void callR(Context* c, unsigned size UNUSED, Assembler::Register* target) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); emit(c, blx(target->low)); } void callC(Context* c, unsigned size UNUSED, Assembler::Constant* target) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); appendOffsetTask(c, target->value, offset(c)); emit(c, bl(0)); @@ -1581,27 +1584,27 @@ callC(Context* c, unsigned size UNUSED, Assembler::Constant* target) void longCallC(Context* c, unsigned size UNUSED, Assembler::Constant* target) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); Assembler::Register tmp(4); - moveCR2(c, BytesPerWord, target, BytesPerWord, &tmp, offset(c)); - callR(c, BytesPerWord, &tmp); + moveCR2(c, TargetBytesPerWord, target, TargetBytesPerWord, &tmp, offset(c)); + callR(c, TargetBytesPerWord, &tmp); } void longJumpC(Context* c, unsigned size UNUSED, Assembler::Constant* target) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); Assembler::Register tmp(4); // a non-arg reg that we don't mind clobbering - moveCR2(c, BytesPerWord, target, BytesPerWord, &tmp, offset(c)); - jumpR(c, BytesPerWord, &tmp); + moveCR2(c, TargetBytesPerWord, target, TargetBytesPerWord, &tmp, offset(c)); + jumpR(c, TargetBytesPerWord, &tmp); } void jumpC(Context* c, unsigned size UNUSED, Assembler::Constant* target) { - assert(c, size == BytesPerWord); + assert(c, size == TargetBytesPerWord); appendOffsetTask(c, target->value, offset(c)); emit(c, b(0)); @@ -1630,7 +1633,7 @@ nextFrame(ArchitectureContext* c, uint32_t* start, unsigned size UNUSED, unsigned targetParameterFootprint UNUSED, void** ip, void** stack) { assert(c, *ip >= start); - assert(c, *ip <= start + (size / BytesPerWord)); + assert(c, *ip <= start + (size / TargetBytesPerWord)); uint32_t* instruction = static_cast(*ip); @@ -1669,7 +1672,7 @@ nextFrame(ArchitectureContext* c, uint32_t* start, unsigned size UNUSED, unsigned value = *instruction & 0xff; unsigned rotation = (*instruction >> 8) & 0xf; switch (rotation) { - case 0: offset -= value / BytesPerWord; break; + case 0: offset -= value / TargetBytesPerWord; break; case 15: offset -= value; break; default: abort(c); } @@ -1905,8 +1908,8 @@ class MyArchitecture: public Assembler::Architecture { return 4; } - virtual void setConstant(void* dst, uintptr_t constant) { - *static_cast(dst) = constant; + virtual void setConstant(void* dst, uint64_t constant) { + *static_cast(dst) = constant; } virtual unsigned alignFrameSize(unsigned sizeInWords) { @@ -2146,18 +2149,18 @@ class MyAssembler: public Assembler { Constant handlerConstant (new (c.zone->allocate(sizeof(ResolvedPromise))) ResolvedPromise(handler)); - branchRM(&c, JumpIfGreaterOrEqual, BytesPerWord, &stack, &stackLimit, + branchRM(&c, JumpIfGreaterOrEqual, TargetBytesPerWord, &stack, &stackLimit, &handlerConstant); } virtual void saveFrame(unsigned stackOffset, unsigned ipOffset) { Register link(LinkRegister); Memory linkDst(ThreadRegister, ipOffset); - moveRM(&c, BytesPerWord, &link, BytesPerWord, &linkDst); + moveRM(&c, TargetBytesPerWord, &link, TargetBytesPerWord, &linkDst); Register stack(StackRegister); Memory stackDst(ThreadRegister, stackOffset); - moveRM(&c, BytesPerWord, &stack, BytesPerWord, &stackDst); + moveRM(&c, TargetBytesPerWord, &stack, TargetBytesPerWord, &stackDst); } virtual void pushFrame(unsigned argumentCount, ...) { @@ -2173,7 +2176,7 @@ class MyAssembler: public Assembler { arguments[i].size = va_arg(a, unsigned); arguments[i].type = static_cast(va_arg(a, int)); arguments[i].operand = va_arg(a, Operand*); - footprint += ceiling(arguments[i].size, BytesPerWord); + footprint += ceiling(arguments[i].size, TargetBytesPerWord); } va_end(a); @@ -2186,17 +2189,18 @@ class MyAssembler: public Assembler { apply(Move, arguments[i].size, arguments[i].type, arguments[i].operand, - pad(arguments[i].size), RegisterOperand, &dst); + pad(arguments[i].size, TargetBytesPerWord), RegisterOperand, + &dst); - offset += ceiling(arguments[i].size, BytesPerWord); + offset += ceiling(arguments[i].size, TargetBytesPerWord); } else { - Memory dst(StackRegister, offset * BytesPerWord); + Memory dst(StackRegister, offset * TargetBytesPerWord); apply(Move, arguments[i].size, arguments[i].type, arguments[i].operand, - pad(arguments[i].size), MemoryOperand, &dst); + pad(arguments[i].size, TargetBytesPerWord), MemoryOperand, &dst); - offset += ceiling(arguments[i].size, BytesPerWord); + offset += ceiling(arguments[i].size, TargetBytesPerWord); } } } @@ -2210,33 +2214,37 @@ class MyAssembler: public Assembler { assert(&c, footprint < 256); Register stack(StackRegister); - ResolvedPromise footprintPromise(footprint * BytesPerWord); + ResolvedPromise footprintPromise(footprint * TargetBytesPerWord); Constant footprintConstant(&footprintPromise); - subC(&c, BytesPerWord, &footprintConstant, &stack, &stack); + subC(&c, TargetBytesPerWord, &footprintConstant, &stack, &stack); Register returnAddress(LinkRegister); - Memory returnAddressDst(StackRegister, (footprint - 1) * BytesPerWord); - moveRM(&c, BytesPerWord, &returnAddress, BytesPerWord, &returnAddressDst); + Memory returnAddressDst + (StackRegister, (footprint - 1) * TargetBytesPerWord); + moveRM(&c, TargetBytesPerWord, &returnAddress, TargetBytesPerWord, + &returnAddressDst); } virtual void adjustFrame(unsigned difference) { Register stack(StackRegister); - ResolvedPromise differencePromise(difference * BytesPerWord); + ResolvedPromise differencePromise(difference * TargetBytesPerWord); Constant differenceConstant(&differencePromise); - subC(&c, BytesPerWord, &differenceConstant, &stack, &stack); + subC(&c, TargetBytesPerWord, &differenceConstant, &stack, &stack); } virtual void popFrame(unsigned footprint) { footprint += FrameHeaderSize; Register returnAddress(LinkRegister); - Memory returnAddressSrc(StackRegister, (footprint - 1) * BytesPerWord); - moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &returnAddress); + Memory returnAddressSrc + (StackRegister, (footprint - 1) * TargetBytesPerWord); + moveMR(&c, TargetBytesPerWord, &returnAddressSrc, TargetBytesPerWord, + &returnAddress); Register stack(StackRegister); - ResolvedPromise footprintPromise(footprint * BytesPerWord); + ResolvedPromise footprintPromise(footprint * TargetBytesPerWord); Constant footprintConstant(&footprintPromise); - addC(&c, BytesPerWord, &footprintConstant, &stack, &stack); + addC(&c, TargetBytesPerWord, &footprintConstant, &stack, &stack); } virtual void popFrameForTailCall(unsigned footprint, @@ -2252,20 +2260,22 @@ class MyAssembler: public Assembler { Register link(LinkRegister); Memory returnAddressSrc - (StackRegister, (footprint - 1) * BytesPerWord); - moveMR(&c, BytesPerWord, &returnAddressSrc, BytesPerWord, &link); + (StackRegister, (footprint - 1) * TargetBytesPerWord); + moveMR(&c, TargetBytesPerWord, &returnAddressSrc, TargetBytesPerWord, + &link); Register stack(StackRegister); - ResolvedPromise footprintPromise((footprint - offset) * BytesPerWord); + ResolvedPromise footprintPromise + ((footprint - offset) * TargetBytesPerWord); Constant footprintConstant(&footprintPromise); - addC(&c, BytesPerWord, &footprintConstant, &stack, &stack); + addC(&c, TargetBytesPerWord, &footprintConstant, &stack, &stack); if (returnAddressSurrogate != NoRegister) { assert(&c, offset > 0); Register ras(returnAddressSurrogate); - Memory dst(StackRegister, (offset - 1) * BytesPerWord); - moveRM(&c, BytesPerWord, &ras, BytesPerWord, &dst); + Memory dst(StackRegister, (offset - 1) * TargetBytesPerWord); + moveRM(&c, TargetBytesPerWord, &ras, TargetBytesPerWord, &dst); } } else { popFrame(footprint); @@ -2288,9 +2298,9 @@ class MyAssembler: public Assembler { offset = argumentFootprint - StackAlignmentInWords; Register stack(StackRegister); - ResolvedPromise adjustmentPromise(offset * BytesPerWord); + ResolvedPromise adjustmentPromise(offset * TargetBytesPerWord); Constant adjustment(&adjustmentPromise); - addC(&c, BytesPerWord, &adjustment, &stack, &stack); + addC(&c, TargetBytesPerWord, &adjustment, &stack, &stack); } else { offset = 0; } @@ -2305,7 +2315,7 @@ class MyAssembler: public Assembler { Register stack(StackRegister); Memory newStackSrc(ThreadRegister, stackOffsetFromThread); - moveMR(&c, BytesPerWord, &newStackSrc, BytesPerWord, &stack); + moveMR(&c, TargetBytesPerWord, &newStackSrc, TargetBytesPerWord, &stack); return_(&c); } @@ -2338,7 +2348,7 @@ class MyAssembler: public Assembler { { if (isBranch(op)) { assert(&c, aSize == bSize); - assert(&c, cSize == BytesPerWord); + assert(&c, cSize == TargetBytesPerWord); assert(&c, cType == ConstantOperand); arch_->c.branchOperations[branchIndex(&(arch_->c), aType, bType)] @@ -2382,7 +2392,7 @@ class MyAssembler: public Assembler { unsigned entry = dstOffset + poolSize; if (needJump(b)) { - entry += BytesPerWord; + entry += TargetBytesPerWord; } o->entry->address = dst + entry; @@ -2396,14 +2406,15 @@ class MyAssembler: public Assembler { int32_t* p = reinterpret_cast(dst + instruction); *p = (v & PoolOffsetMask) | ((~PoolOffsetMask) & *p); - poolSize += BytesPerWord; + poolSize += TargetBytesPerWord; } if (needJump(b)) { - write4(dst + dstOffset, ::b((poolSize + BytesPerWord - 8) >> 2)); + write4 + (dst + dstOffset, ::b((poolSize + TargetBytesPerWord - 8) >> 2)); } - dstOffset += poolSize + BytesPerWord; + dstOffset += poolSize + TargetBytesPerWord; } unsigned size = b->size - blockOffset; @@ -2421,10 +2432,10 @@ class MyAssembler: public Assembler { for (ConstantPoolEntry* e = c.constantPool; e; e = e->next) { if (e->constant->resolved()) { - *static_cast(e->address) = e->constant->value(); + *static_cast(e->address) = e->constant->value(); } else { new (e->constant->listen(sizeof(ConstantPoolListener))) - ConstantPoolListener(c.s, static_cast(e->address), + ConstantPoolListener(c.s, static_cast(e->address), e->callOffset ? dst + e->callOffset->value() + 8 : 0); @@ -2453,7 +2464,7 @@ class MyAssembler: public Assembler { MyBlock* b = c.lastBlock; unsigned thisEventOffset = c.code.length() - b->offset; if (b->poolOffsetHead) { - int32_t v = (thisEventOffset + BytesPerWord - 8) + int32_t v = (thisEventOffset + TargetBytesPerWord - 8) - b->poolOffsetHead->offset; if (v > 0 and v != (v & PoolOffsetMask)) {