|
@@ -17,6 +17,7 @@
|
|
|
|
|
|
#define DT_OPERATION(op) \
|
|
#define DT_OPERATION(op) \
|
|
case DT_INT: cAddOperation(OP_##op##_INT); break; \
|
|
case DT_INT: cAddOperation(OP_##op##_INT); break; \
|
|
|
|
+ case DT_LONG: cAddOperation(OP_##op##_LONG); break; \
|
|
case DT_BOOL: cAddOperation(OP_##op##_BOOL); break; \
|
|
case DT_BOOL: cAddOperation(OP_##op##_BOOL); break; \
|
|
case DT_FLOAT: cAddOperation(OP_##op##_FLOAT); break;
|
|
case DT_FLOAT: cAddOperation(OP_##op##_FLOAT); break;
|
|
|
|
|
|
@@ -47,31 +48,38 @@ static int continueAt = 0;
|
|
|
|
|
|
typedef struct {
|
|
typedef struct {
|
|
Operation intOp;
|
|
Operation intOp;
|
|
|
|
+ Operation longOp;
|
|
Operation floatOp;
|
|
Operation floatOp;
|
|
Operation boolOp;
|
|
Operation boolOp;
|
|
Operation pointerOp;
|
|
Operation pointerOp;
|
|
const char* name;
|
|
const char* name;
|
|
} TypedOp;
|
|
} TypedOp;
|
|
|
|
|
|
-#define TYPE_OP(NAME, INT, FLOAT, BOOL, POINTER, text) \
|
|
|
|
- static const TypedOp TYPED_##NAME = {OP_##INT, OP_##FLOAT, OP_##BOOL, \
|
|
|
|
- OP_##POINTER, text};
|
|
|
|
-TYPE_OP(MUL, MUL_INT, MUL_FLOAT, NOTHING, NOTHING, "*")
|
|
|
|
-TYPE_OP(DIV, DIV_INT, DIV_FLOAT, NOTHING, NOTHING, "/")
|
|
|
|
-TYPE_OP(MOD, MOD_INT, NOTHING, NOTHING, NOTHING, "%")
|
|
|
|
-TYPE_OP(ADD, ADD_INT, ADD_FLOAT, NOTHING, NOTHING, "+")
|
|
|
|
-TYPE_OP(SUB, SUB_INT, SUB_FLOAT, NOTHING, NOTHING, "-")
|
|
|
|
-TYPE_OP(LESS, LESS_INT, LESS_FLOAT, NOTHING, NOTHING, "<")
|
|
|
|
-TYPE_OP(LESS_EQUAL, GREATER_INT, GREATER_FLOAT, NOTHING, NOTHING, "<=")
|
|
|
|
-TYPE_OP(GREATER, GREATER_INT, GREATER_FLOAT, NOTHING, NOTHING, ">")
|
|
|
|
-TYPE_OP(GREATER_EQUAL, LESS_INT, LESS_FLOAT, NOTHING, NOTHING, ">=")
|
|
|
|
-TYPE_OP(EQUAL, EQUAL_INT, EQUAL_FLOAT, EQUAL_BOOL, EQUAL_POINTER, "==")
|
|
|
|
-TYPE_OP(NOT_EQUAL, EQUAL_INT, EQUAL_FLOAT, EQUAL_BOOL, EQUAL_POINTER, "!=")
|
|
|
|
-TYPE_OP(BIT_OR, BIT_OR, NOTHING, NOTHING, NOTHING, "|")
|
|
|
|
-TYPE_OP(BIT_XOR, BIT_XOR, NOTHING, NOTHING, NOTHING, "^")
|
|
|
|
-TYPE_OP(BIT_AND, BIT_AND, NOTHING, NOTHING, NOTHING, "&")
|
|
|
|
-TYPE_OP(LEFT_SHIFT, LEFT_SHIFT, NOTHING, NOTHING, NOTHING, "<<")
|
|
|
|
-TYPE_OP(RIGHT_SHIFT, RIGHT_SHIFT, NOTHING, NOTHING, NOTHING, ">>")
|
|
|
|
|
|
+#define TYPE_OP(NAME, INT, LONG, FLOAT, BOOL, POINTER, text) \
|
|
|
|
+ static const TypedOp TYPED_##NAME = {OP_##INT, OP_##LONG, OP_##FLOAT, \
|
|
|
|
+ OP_##BOOL, OP_##POINTER, text};
|
|
|
|
+TYPE_OP(MUL, MUL_INT, MUL_LONG, MUL_FLOAT, NOTHING, NOTHING, "*")
|
|
|
|
+TYPE_OP(DIV, DIV_INT, DIV_LONG, DIV_FLOAT, NOTHING, NOTHING, "/")
|
|
|
|
+TYPE_OP(MOD, MOD_INT, MOD_LONG, NOTHING, NOTHING, NOTHING, "%")
|
|
|
|
+TYPE_OP(ADD, ADD_INT, ADD_LONG, ADD_FLOAT, NOTHING, NOTHING, "+")
|
|
|
|
+TYPE_OP(SUB, SUB_INT, SUB_LONG, SUB_FLOAT, NOTHING, NOTHING, "-")
|
|
|
|
+TYPE_OP(LESS, LESS_INT, LESS_LONG, LESS_FLOAT, NOTHING, NOTHING, "<")
|
|
|
|
+TYPE_OP(LESS_EQUAL, GREATER_INT, GREATER_LONG, GREATER_FLOAT, NOTHING, NOTHING,
|
|
|
|
+ "<=")
|
|
|
|
+TYPE_OP(GREATER, GREATER_INT, GREATER_LONG, GREATER_FLOAT, NOTHING, NOTHING,
|
|
|
|
+ ">")
|
|
|
|
+TYPE_OP(GREATER_EQUAL, LESS_INT, LESS_LONG, LESS_FLOAT, NOTHING, NOTHING, ">=")
|
|
|
|
+TYPE_OP(EQUAL, EQUAL_INT, EQUAL_LONG, EQUAL_FLOAT, EQUAL_BOOL, EQUAL_POINTER,
|
|
|
|
+ "==")
|
|
|
|
+TYPE_OP(NOT_EQUAL, EQUAL_INT, EQUAL_LONG, EQUAL_FLOAT, EQUAL_BOOL,
|
|
|
|
+ EQUAL_POINTER, "!=")
|
|
|
|
+TYPE_OP(BIT_OR, BIT_OR_INT, BIT_OR_LONG, NOTHING, NOTHING, NOTHING, "|")
|
|
|
|
+TYPE_OP(BIT_XOR, BIT_XOR_INT, BIT_XOR_LONG, NOTHING, NOTHING, NOTHING, "^")
|
|
|
|
+TYPE_OP(BIT_AND, BIT_AND_INT, BIT_AND_LONG, NOTHING, NOTHING, NOTHING, "&")
|
|
|
|
+TYPE_OP(LEFT_SHIFT, LEFT_SHIFT_INT, LEFT_SHIFT_LONG, NOTHING, NOTHING, NOTHING,
|
|
|
|
+ "<<")
|
|
|
|
+TYPE_OP(RIGHT_SHIFT, RIGHT_SHIFT_INT, RIGHT_SHIFT_LONG, NOTHING, NOTHING,
|
|
|
|
+ NOTHING, ">>")
|
|
|
|
|
|
static void cError(const char* format, ...) {
|
|
static void cError(const char* format, ...) {
|
|
va_list args;
|
|
va_list args;
|
|
@@ -122,6 +130,10 @@ static void cAddInt(int i) {
|
|
bcAddBytes(code, &i, sizeof(int));
|
|
bcAddBytes(code, &i, sizeof(int));
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void cAddLong(long l) {
|
|
|
|
+ bcAddBytes(code, &l, sizeof(long));
|
|
|
|
+}
|
|
|
|
+
|
|
static void cAddIntOperation(Operation token, int i) {
|
|
static void cAddIntOperation(Operation token, int i) {
|
|
cAddOperation(token);
|
|
cAddOperation(token);
|
|
cAddInt(i);
|
|
cAddInt(i);
|
|
@@ -165,6 +177,15 @@ static void cConstantInt() {
|
|
cAddIntOperation(OP_PUSH_INT, value);
|
|
cAddIntOperation(OP_PUSH_INT, value);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static void cConstantLong() {
|
|
|
|
+ long value;
|
|
|
|
+ if(!tReadLong(&value)) {
|
|
|
|
+ cError("long token without an long");
|
|
|
|
+ }
|
|
|
|
+ cAddOperation(OP_PUSH_LONG);
|
|
|
|
+ cAddLong(value);
|
|
|
|
+}
|
|
|
|
+
|
|
static void cConstantFloat() {
|
|
static void cConstantFloat() {
|
|
float value;
|
|
float value;
|
|
if(!tReadFloat(&value)) {
|
|
if(!tReadFloat(&value)) {
|
|
@@ -203,6 +224,7 @@ static DataType cExtendType(DataType dt) {
|
|
static DataType cReadType(Token t) {
|
|
static DataType cReadType(Token t) {
|
|
switch(t) {
|
|
switch(t) {
|
|
case T_INT: return cExtendType(dtInt());
|
|
case T_INT: return cExtendType(dtInt());
|
|
|
|
+ case T_LONG: return cExtendType(dtLong());
|
|
case T_BOOL: return cExtendType(dtBool());
|
|
case T_BOOL: return cExtendType(dtBool());
|
|
case T_FLOAT: return cExtendType(dtFloat());
|
|
case T_FLOAT: return cExtendType(dtFloat());
|
|
case T_LITERAL:
|
|
case T_LITERAL:
|
|
@@ -340,6 +362,7 @@ static DataType cPrimary() {
|
|
Token t = cReadTokenAndLine();
|
|
Token t = cReadTokenAndLine();
|
|
switch(t) {
|
|
switch(t) {
|
|
case T_CONST_INT: cConstantInt(); return dtInt();
|
|
case T_CONST_INT: cConstantInt(); return dtInt();
|
|
|
|
+ case T_CONST_LONG: cConstantLong(); return dtLong();
|
|
case T_CONST_FLOAT: cConstantFloat(); return dtFloat();
|
|
case T_CONST_FLOAT: cConstantFloat(); return dtFloat();
|
|
case T_TRUE: cAddOperation(OP_PUSH_TRUE); return dtBool();
|
|
case T_TRUE: cAddOperation(OP_PUSH_TRUE); return dtBool();
|
|
case T_FALSE: cAddOperation(OP_PUSH_FALSE); return dtBool();
|
|
case T_FALSE: cAddOperation(OP_PUSH_FALSE); return dtBool();
|
|
@@ -363,14 +386,24 @@ static void cExpectType(DataType actual, DataType wanted, const char* name) {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-static void cPostChange(DataType* dt, int change, const char* name) {
|
|
|
|
- cRemoveReference(dt, name);
|
|
|
|
- cExpectType(*dt, dtInt(), name);
|
|
|
|
|
|
+static void cChangeType(DataType* dt, Operation op, Operation pushOp,
|
|
|
|
+ int change) {
|
|
if(onLine) {
|
|
if(onLine) {
|
|
- cAddByteOperation(OP_INT_CHANGE, change);
|
|
|
|
|
|
+ cAddByteOperation(op, change);
|
|
*dt = dtVoid();
|
|
*dt = dtVoid();
|
|
} else {
|
|
} else {
|
|
- cAddByteOperation(OP_PUSH_POST_INT_CHANGE, change);
|
|
|
|
|
|
+ cAddByteOperation(pushOp, change);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static void cPostChange(DataType* dt, int change, const char* name) {
|
|
|
|
+ cRemoveReference(dt, name);
|
|
|
|
+ if(dtCompare(*dt, dtInt())) {
|
|
|
|
+ cChangeType(dt, OP_CHANGE_INT, OP_PUSH_POST_CHANGE_INT, change);
|
|
|
|
+ } else if(dtCompare(*dt, dtLong())) {
|
|
|
|
+ cChangeType(dt, OP_CHANGE_LONG, OP_PUSH_POST_CHANGE_LONG, change);
|
|
|
|
+ } else {
|
|
|
|
+ cError("%s needs an int or long not %s", name, cGetName(*dt));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
@@ -421,18 +454,21 @@ static DataType cAccess() {
|
|
|
|
|
|
static DataType cPreChange(DataType dt, int change, const char* name) {
|
|
static DataType cPreChange(DataType dt, int change, const char* name) {
|
|
cRemoveReference(&dt, name);
|
|
cRemoveReference(&dt, name);
|
|
- cExpectType(dt, dtInt(), name);
|
|
|
|
- if(onLine) {
|
|
|
|
- cAddByteOperation(OP_INT_CHANGE, change);
|
|
|
|
- return dtVoid();
|
|
|
|
|
|
+ if(dtCompare(dt, dtInt())) {
|
|
|
|
+ cChangeType(&dt, OP_CHANGE_INT, OP_PUSH_PRE_CHANGE_INT, change);
|
|
|
|
+ } else if(dtCompare(dt, dtLong())) {
|
|
|
|
+ cChangeType(&dt, OP_CHANGE_LONG, OP_PUSH_PRE_CHANGE_LONG, change);
|
|
|
|
+ } else {
|
|
|
|
+ cError("%s needs an int or long not %s", name, cGetName(dt));
|
|
}
|
|
}
|
|
- cAddByteOperation(OP_PUSH_PRE_INT_CHANGE, change);
|
|
|
|
return dt;
|
|
return dt;
|
|
}
|
|
}
|
|
|
|
|
|
static DataType cInvertSign(DataType dt) {
|
|
static DataType cInvertSign(DataType dt) {
|
|
if(dtIsInt(dt)) {
|
|
if(dtIsInt(dt)) {
|
|
cAddOperation(OP_INVERT_SIGN_INT);
|
|
cAddOperation(OP_INVERT_SIGN_INT);
|
|
|
|
+ } else if(dtIsLong(dt)) {
|
|
|
|
+ cAddOperation(OP_INVERT_SIGN_LONG);
|
|
} else if(dtIsFloat(dt)) {
|
|
} else if(dtIsFloat(dt)) {
|
|
cAddOperation(OP_INVERT_SIGN_FLOAT);
|
|
cAddOperation(OP_INVERT_SIGN_FLOAT);
|
|
} else {
|
|
} else {
|
|
@@ -441,25 +477,30 @@ static DataType cInvertSign(DataType dt) {
|
|
return dt;
|
|
return dt;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+static DataType cCast(DataType in, DataType a, Operation aOp, DataType b,
|
|
|
|
+ Operation bOp, DataType out) {
|
|
|
|
+ if(dtCompare(in, a)) {
|
|
|
|
+ cAddOperation(aOp);
|
|
|
|
+ } else if(dtCompare(in, b)) {
|
|
|
|
+ cAddOperation(bOp);
|
|
|
|
+ } else {
|
|
|
|
+ cError("cannot cast %s to %s", cGetName(in), cGetName(out));
|
|
|
|
+ }
|
|
|
|
+ return out;
|
|
|
|
+}
|
|
|
|
+
|
|
static DataType cPreUnary() {
|
|
static DataType cPreUnary() {
|
|
int marker = tGetMarker();
|
|
int marker = tGetMarker();
|
|
if(cConsumeTokenIf(T_OPEN_BRACKET)) {
|
|
if(cConsumeTokenIf(T_OPEN_BRACKET)) {
|
|
if(cConsumeTokenIf(T_FLOAT) && cConsumeTokenIf(T_CLOSE_BRACKET)) {
|
|
if(cConsumeTokenIf(T_FLOAT) && cConsumeTokenIf(T_CLOSE_BRACKET)) {
|
|
- DataType dt = cUnpack(cPreUnary());
|
|
|
|
- if(dtCompare(dt, dtInt())) {
|
|
|
|
- cAddOperation(OP_INT_TO_FLOAT);
|
|
|
|
- } else {
|
|
|
|
- cError("cannot cast %s to float", cGetName(dt));
|
|
|
|
- }
|
|
|
|
- return dtFloat();
|
|
|
|
|
|
+ return cCast(cUnpack(cPreUnary()), dtInt(), OP_INT_TO_FLOAT,
|
|
|
|
+ dtLong(), OP_LONG_TO_FLOAT, dtFloat());
|
|
} else if(cConsumeTokenIf(T_INT) && cConsumeTokenIf(T_CLOSE_BRACKET)) {
|
|
} else if(cConsumeTokenIf(T_INT) && cConsumeTokenIf(T_CLOSE_BRACKET)) {
|
|
- DataType dt = cUnpack(cPreUnary());
|
|
|
|
- if(dtCompare(dt, dtFloat())) {
|
|
|
|
- cAddOperation(OP_FLOAT_TO_INT);
|
|
|
|
- } else {
|
|
|
|
- cError("cannot cast %s to int", cGetName(dt));
|
|
|
|
- }
|
|
|
|
- return dtInt();
|
|
|
|
|
|
+ return cCast(cUnpack(cPreUnary()), dtFloat(), OP_FLOAT_TO_INT,
|
|
|
|
+ dtLong(), OP_LONG_TO_INT, dtInt());
|
|
|
|
+ } else if(cConsumeTokenIf(T_LONG) && cConsumeTokenIf(T_CLOSE_BRACKET)) {
|
|
|
|
+ return cCast(cUnpack(cPreUnary()), dtInt(), OP_INT_TO_LONG,
|
|
|
|
+ dtFloat(), OP_FLOAT_TO_LONG, dtLong());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
tReset(marker);
|
|
tReset(marker);
|
|
@@ -476,8 +517,13 @@ static DataType cPreUnary() {
|
|
return dt;
|
|
return dt;
|
|
} else if(cConsumeTokenIf(T_BIT_NOT)) {
|
|
} else if(cConsumeTokenIf(T_BIT_NOT)) {
|
|
DataType dt = cPreUnary();
|
|
DataType dt = cPreUnary();
|
|
- cExpectType(dt, dtInt(), "!");
|
|
|
|
- cAddOperation(OP_BIT_NOT);
|
|
|
|
|
|
+ if(dtCompare(dt, dtInt())) {
|
|
|
|
+ cAddOperation(OP_BIT_NOT_INT);
|
|
|
|
+ } else if(dtCompare(dt, dtLong())) {
|
|
|
|
+ cAddOperation(OP_BIT_NOT_LONG);
|
|
|
|
+ } else {
|
|
|
|
+ cError("~ needs an int or long not %s", cGetName(dt));
|
|
|
|
+ }
|
|
return dt;
|
|
return dt;
|
|
} else if(cConsumeTokenIf(T_BIT_AND)) {
|
|
} else if(cConsumeTokenIf(T_BIT_AND)) {
|
|
DataType dt = cPreUnary();
|
|
DataType dt = cPreUnary();
|
|
@@ -506,6 +552,8 @@ static void cAddTypeOperation(DataType* a, Parser bf, const TypedOp* op) {
|
|
cInvalidOperation(*a, b, op->name);
|
|
cInvalidOperation(*a, b, op->name);
|
|
} else if(dtIsInt(*a) && op->intOp != OP_NOTHING) {
|
|
} else if(dtIsInt(*a) && op->intOp != OP_NOTHING) {
|
|
cAddOperation(op->intOp);
|
|
cAddOperation(op->intOp);
|
|
|
|
+ } else if(dtIsLong(*a) && op->longOp != OP_NOTHING) {
|
|
|
|
+ cAddOperation(op->longOp);
|
|
} else if(dtIsFloat(*a) && op->floatOp != OP_NOTHING) {
|
|
} else if(dtIsFloat(*a) && op->floatOp != OP_NOTHING) {
|
|
cAddOperation(op->floatOp);
|
|
cAddOperation(op->floatOp);
|
|
} else if(dtIsBool(*a) && op->boolOp != OP_NOTHING) {
|
|
} else if(dtIsBool(*a) && op->boolOp != OP_NOTHING) {
|
|
@@ -681,6 +729,8 @@ static void cReturn() {
|
|
cError("wrong return type, should be %s", cGetName(returnType));
|
|
cError("wrong return type, should be %s", cGetName(returnType));
|
|
} else if(dtIsInt(dt)) {
|
|
} else if(dtIsInt(dt)) {
|
|
cAddReturn(OP_RETURN_INT);
|
|
cAddReturn(OP_RETURN_INT);
|
|
|
|
+ } else if(dtIsLong(dt)) {
|
|
|
|
+ cAddReturn(OP_RETURN_LONG);
|
|
} else if(dtIsBool(dt)) {
|
|
} else if(dtIsBool(dt)) {
|
|
cAddReturn(OP_RETURN_BOOL);
|
|
cAddReturn(OP_RETURN_BOOL);
|
|
} else if(dtIsFloat(dt)) {
|
|
} else if(dtIsFloat(dt)) {
|
|
@@ -826,6 +876,7 @@ static void cLineExpression() {
|
|
}
|
|
}
|
|
switch(t) {
|
|
switch(t) {
|
|
case T_INT: cDeclare(dtInt()); break;
|
|
case T_INT: cDeclare(dtInt()); break;
|
|
|
|
+ case T_LONG: cDeclare(dtLong()); break;
|
|
case T_BOOL: cDeclare(dtBool()); break;
|
|
case T_BOOL: cDeclare(dtBool()); break;
|
|
case T_FLOAT: cDeclare(dtFloat()); break;
|
|
case T_FLOAT: cDeclare(dtFloat()); break;
|
|
case T_DELETE: cDelete(); break;
|
|
case T_DELETE: cDelete(); break;
|