modify (THREAD_)RUNTIME_ARRAY definition so RUNTIME_ARRAY_BODY must be used

Previously, if you forgot to use RUNTIME_ARRAY_BODY to reference an
array declared with (THREAD_)RUNTIME_ARRAY, you wouldn't get a
compiler error until you tried to build on e.g. MSVC, where
runtime-sized stack arrays aren't supported.  This change ensures you
find out regardless of what compiler you're using, which ought to
protect us from regressions going forward.
This commit is contained in:
Joel Dice 2013-02-20 17:20:17 -07:00
parent fd047bd6e9
commit f04f444f23
5 changed files with 49 additions and 44 deletions

View File

@ -32,9 +32,9 @@ class RuntimeArray {
#else // not _MSC_VER #else // not _MSC_VER
# define RUNTIME_ARRAY(type, name, size) type name[size]; # define RUNTIME_ARRAY(type, name, size) type name##_body[size];
# define RUNTIME_ARRAY_BODY(name) name # define RUNTIME_ARRAY_BODY(name) name##_body
#endif #endif
#endif // AVIAN_UTIL_RUNTIME_ARRAY_H #endif // AVIAN_UTIL_RUNTIME_ARRAY_H

View File

@ -18,6 +18,7 @@
#include <avian/vm/codegen/promise.h> #include <avian/vm/codegen/promise.h>
#include "target.h" #include "target.h"
#include <avian/tools/object-writer/tools.h> #include <avian/tools/object-writer/tools.h>
#include <avian/util/runtime-array.h>
#include "lzma.h" #include "lzma.h"
// since we aren't linking against libstdc++, we must implement this // since we aren't linking against libstdc++, we must implement this
@ -345,20 +346,20 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
unsigned count = s.read2() - 1; unsigned count = s.read2() - 1;
if (count) { if (count) {
THREAD_RUNTIME_ARRAY(t, Type, types, count + 2); THREAD_RUNTIME_ARRAY(t, Type, types, count + 2);
types[0] = Type_object; RUNTIME_ARRAY_BODY(types)[0] = Type_object;
types[1] = Type_intptr_t; RUNTIME_ARRAY_BODY(types)[1] = Type_intptr_t;
for (unsigned i = 2; i < count + 2; ++i) { for (unsigned i = 2; i < count + 2; ++i) {
switch (s.read1()) { switch (s.read1()) {
case CONSTANT_Class: case CONSTANT_Class:
case CONSTANT_String: case CONSTANT_String:
types[i] = Type_object; RUNTIME_ARRAY_BODY(types)[i] = Type_object;
s.skip(2); s.skip(2);
break; break;
case CONSTANT_Integer: case CONSTANT_Integer:
case CONSTANT_Float: case CONSTANT_Float:
types[i] = Type_int32_t; RUNTIME_ARRAY_BODY(types)[i] = Type_int32_t;
s.skip(4); s.skip(4);
break; break;
@ -366,24 +367,24 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
case CONSTANT_Fieldref: case CONSTANT_Fieldref:
case CONSTANT_Methodref: case CONSTANT_Methodref:
case CONSTANT_InterfaceMethodref: case CONSTANT_InterfaceMethodref:
types[i] = Type_object; RUNTIME_ARRAY_BODY(types)[i] = Type_object;
s.skip(4); s.skip(4);
break; break;
case CONSTANT_Long: case CONSTANT_Long:
types[i++] = Type_int64_t; RUNTIME_ARRAY_BODY(types)[i++] = Type_int64_t;
types[i] = Type_int64_t_pad; RUNTIME_ARRAY_BODY(types)[i] = Type_int64_t_pad;
s.skip(8); s.skip(8);
break; break;
case CONSTANT_Double: case CONSTANT_Double:
types[i++] = Type_double; RUNTIME_ARRAY_BODY(types)[i++] = Type_double;
types[i] = Type_double_pad; RUNTIME_ARRAY_BODY(types)[i] = Type_double_pad;
s.skip(8); s.skip(8);
break; break;
case CONSTANT_Utf8: case CONSTANT_Utf8:
types[i] = Type_object; RUNTIME_ARRAY_BODY(types)[i] = Type_object;
s.skip(s.read2()); s.skip(s.read2());
break; break;
@ -403,7 +404,7 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
map->targetFixedOffsets()[i * BytesPerWord] map->targetFixedOffsets()[i * BytesPerWord]
= i * TargetBytesPerWord; = i * TargetBytesPerWord;
init(new (map->fixedFields() + i) Field, types[i], init(new (map->fixedFields() + i) Field, RUNTIME_ARRAY_BODY(types)[i],
i * BytesPerWord, BytesPerWord, i * TargetBytesPerWord, i * BytesPerWord, BytesPerWord, i * TargetBytesPerWord,
TargetBytesPerWord); TargetBytesPerWord);
} }
@ -439,15 +440,15 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
for (unsigned j = 0; j < map->fixedFieldCount; ++j) { for (unsigned j = 0; j < map->fixedFieldCount; ++j) {
Field* f = map->fixedFields() + j; Field* f = map->fixedFields() + j;
memberFields[memberIndex] = *f; RUNTIME_ARRAY_BODY(memberFields)[memberIndex] = *f;
targetMemberOffset = f->targetOffset + f->targetSize; targetMemberOffset = f->targetOffset + f->targetSize;
++ memberIndex; ++ memberIndex;
} }
} else { } else {
init(new (&memberFields[0]) Field, Type_object, 0, BytesPerWord, 0, init(new (RUNTIME_ARRAY_BODY(memberFields)) Field, Type_object, 0,
TargetBytesPerWord); BytesPerWord, 0, TargetBytesPerWord);
memberIndex = 1; memberIndex = 1;
buildMemberOffset = BytesPerWord; buildMemberOffset = BytesPerWord;
@ -458,14 +459,16 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
THREAD_RUNTIME_ARRAY(t, Field, staticFields, count + StaticHeader); THREAD_RUNTIME_ARRAY(t, Field, staticFields, count + StaticHeader);
init(new (&staticFields[0]) Field, Type_object, 0, BytesPerWord, 0, init(new (RUNTIME_ARRAY_BODY(staticFields)) Field, Type_object, 0,
BytesPerWord, 0, TargetBytesPerWord);
init(new (RUNTIME_ARRAY_BODY(staticFields) + 1) Field, Type_intptr_t,
BytesPerWord, BytesPerWord, TargetBytesPerWord,
TargetBytesPerWord); TargetBytesPerWord);
init(new (&staticFields[1]) Field, Type_intptr_t, BytesPerWord, init(new (RUNTIME_ARRAY_BODY(staticFields) + 2) Field, Type_object,
BytesPerWord, TargetBytesPerWord, TargetBytesPerWord); BytesPerWord * 2, BytesPerWord, TargetBytesPerWord * 2,
TargetBytesPerWord);
init(new (&staticFields[2]) Field, Type_object, BytesPerWord * 2,
BytesPerWord, TargetBytesPerWord * 2, TargetBytesPerWord);
unsigned staticIndex = StaticHeader; unsigned staticIndex = StaticHeader;
unsigned buildStaticOffset = BytesPerWord * StaticHeader; unsigned buildStaticOffset = BytesPerWord * StaticHeader;
@ -514,8 +517,8 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
buildStaticOffset = fieldOffset(t, field); buildStaticOffset = fieldOffset(t, field);
init(new (&staticFields[staticIndex]) Field, type, init(new (RUNTIME_ARRAY_BODY(staticFields) + staticIndex) Field,
buildStaticOffset, buildSize, targetStaticOffset, type, buildStaticOffset, buildSize, targetStaticOffset,
targetSize); targetSize);
targetStaticOffset += targetSize; targetStaticOffset += targetSize;
@ -528,8 +531,8 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
buildMemberOffset = fieldOffset(t, field); buildMemberOffset = fieldOffset(t, field);
init(new (&memberFields[memberIndex]) Field, type, init(new (RUNTIME_ARRAY_BODY(memberFields) + memberIndex) Field,
buildMemberOffset, buildSize, targetMemberOffset, type, buildMemberOffset, buildSize, targetMemberOffset,
targetSize); targetSize);
targetMemberOffset += targetSize; targetMemberOffset += targetSize;
@ -551,7 +554,7 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
ceilingDivide(targetMemberOffset, TargetBytesPerWord), memberIndex); ceilingDivide(targetMemberOffset, TargetBytesPerWord), memberIndex);
for (unsigned i = 0; i < memberIndex; ++i) { for (unsigned i = 0; i < memberIndex; ++i) {
Field* f = &memberFields[i]; Field* f = RUNTIME_ARRAY_BODY(memberFields) + i;
expect(t, f->buildOffset expect(t, f->buildOffset
< map->buildFixedSizeInWords * BytesPerWord); < map->buildFixedSizeInWords * BytesPerWord);
@ -575,7 +578,7 @@ makeCodeImage(Thread* t, Zone* zone, BootImage* image, uint8_t* code,
TypeMap::SingletonKind); TypeMap::SingletonKind);
for (unsigned i = 0; i < staticIndex; ++i) { for (unsigned i = 0; i < staticIndex; ++i) {
Field* f = &staticFields[i]; Field* f = RUNTIME_ARRAY_BODY(staticFields) + i;
expect(t, f->buildOffset expect(t, f->buildOffset
< map->buildFixedSizeInWords * BytesPerWord); < map->buildFixedSizeInWords * BytesPerWord);
@ -1338,8 +1341,8 @@ writeBootImage2(Thread* t, OutputStream* bootimageOutput, OutputStream* codeOutp
THREAD_RUNTIME_ARRAY(t, Field, fields, count); THREAD_RUNTIME_ARRAY(t, Field, fields, count);
init(new (&fields[0]) Field, Type_object, 0, BytesPerWord, 0, init(new (RUNTIME_ARRAY_BODY(fields)) Field, Type_object, 0,
TargetBytesPerWord); BytesPerWord, 0, TargetBytesPerWord);
unsigned buildOffset = BytesPerWord; unsigned buildOffset = BytesPerWord;
unsigned targetOffset = TargetBytesPerWord; unsigned targetOffset = TargetBytesPerWord;
@ -1416,8 +1419,8 @@ writeBootImage2(Thread* t, OutputStream* bootimageOutput, OutputStream* codeOutp
++ targetOffset; ++ targetOffset;
} }
init(new (&fields[j]) Field, type, buildOffset, buildSize, init(new (RUNTIME_ARRAY_BODY(fields) + j) Field, type, buildOffset,
targetOffset, targetSize); buildSize, targetOffset, targetSize);
buildOffset += buildSize; buildOffset += buildSize;
targetOffset += targetSize; targetOffset += targetSize;
@ -1451,7 +1454,7 @@ writeBootImage2(Thread* t, OutputStream* bootimageOutput, OutputStream* codeOutp
targetArrayElementSize, arrayElementType); targetArrayElementSize, arrayElementType);
for (unsigned j = 0; j < fixedFieldCount; ++j) { for (unsigned j = 0; j < fixedFieldCount; ++j) {
Field* f = &fields[j]; Field* f = RUNTIME_ARRAY_BODY(fields) + j;
expect(t, f->buildOffset expect(t, f->buildOffset
< map->buildFixedSizeInWords * BytesPerWord); < map->buildFixedSizeInWords * BytesPerWord);

View File

@ -2325,19 +2325,20 @@ interpret3(Thread* t, const int base)
THREAD_RUNTIME_ARRAY(t, int32_t, counts, dimensions); THREAD_RUNTIME_ARRAY(t, int32_t, counts, dimensions);
for (int i = dimensions - 1; i >= 0; --i) { for (int i = dimensions - 1; i >= 0; --i) {
counts[i] = popInt(t); RUNTIME_ARRAY_BODY(counts)[i] = popInt(t);
if (UNLIKELY(counts[i] < 0)) { if (UNLIKELY(RUNTIME_ARRAY_BODY(counts)[i] < 0)) {
exception = makeThrowable exception = makeThrowable
(t, Machine::NegativeArraySizeExceptionType, "%d", counts[i]); (t, Machine::NegativeArraySizeExceptionType, "%d",
RUNTIME_ARRAY_BODY(counts)[i]);
goto throw_; goto throw_;
} }
} }
object array = makeArray(t, counts[0]); object array = makeArray(t, RUNTIME_ARRAY_BODY(counts)[0]);
setObjectClass(t, array, class_); setObjectClass(t, array, class_);
PROTECT(t, array); PROTECT(t, array);
populateMultiArray(t, array, counts, 0, dimensions); populateMultiArray(t, array, RUNTIME_ARRAY_BODY(counts), 0, dimensions);
pushObject(t, array); pushObject(t, array);
} goto loop; } goto loop;

View File

@ -4777,13 +4777,13 @@ logTrace(FILE* f, const char* fmt, ...)
RUNTIME_ARRAY(char, buffer, length + 1); RUNTIME_ARRAY(char, buffer, length + 1);
va_start(a, fmt); va_start(a, fmt);
vsnprintf(&buffer[0], length + 1, fmt, a); vsnprintf(RUNTIME_ARRAY_BODY(buffer), length + 1, fmt, a);
va_end(a); va_end(a);
buffer[length] = 0; RUNTIME_ARRAY_BODY(buffer)[length] = 0;
::fprintf(f, "%s", &buffer[0]); ::fprintf(f, "%s", RUNTIME_ARRAY_BODY(buffer));
#ifdef PLATFORM_WINDOWS #ifdef PLATFORM_WINDOWS
::OutputDebugStringA(&buffer[0]); ::OutputDebugStringA(RUNTIME_ARRAY_BODY(buffer));
#endif #endif
} }

View File

@ -1603,7 +1603,8 @@ class ThreadRuntimeArray: public Thread::Resource {
#else // not _MSC_VER #else // not _MSC_VER
# define THREAD_RUNTIME_ARRAY(thread, type, name, size) type name[size]; # define THREAD_RUNTIME_ARRAY(thread, type, name, size) \
type name##_body[size];
#endif // not _MSC_VER #endif // not _MSC_VER