123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 |
- import { Modes } from '../modes';
- import { Types } from "../../typeSystem/types";
- import { StoreObject } from './storeObject';
- import { IType } from '../../typeSystem/itype';
- import { StoreObjectRef } from './storeObjectRef';
- import { ArrayType } from '../../typeSystem/array_type';
- import { ArrayStoreValue } from './value/array_store_value';
- import { Location } from '../../memory/location';
- import { StoreObjectArray } from './storeObjectArray';
- import { IStoreValue } from './value/istore_value';
- import { StoreValue } from './value/store_value';
- import { StoreValueAddress } from './value/store_value_address';
- import { StoreValueRef } from './value/store_value_ref';
- import { ArrayStoreValueRef } from './value/array_store_value_ref';
- import { StoreObjectArrayRef } from './store_object_array_ref';
- export class Store {
- static canImplicitTypeCast (castType: IType, sourceType: IType) {
- if (castType.isCompatible(Types.INTEGER) || castType.isCompatible(Types.REAL)) {
- if (sourceType.isCompatible(Types.INTEGER) || sourceType.isCompatible(Types.REAL)) {
- return true;
- }
- }
- return false;
- }
- static doImplicitCasting (castType: IType, stoObj: IStoreValue) {
- if(!Store.canImplicitTypeCast(castType, stoObj.type)) {
- throw new Error("!!!Critical error: attempted to type cast invalid types");
- }
- if(castType.isCompatible(Types.INTEGER)) {
- return new StoreValue(Types.INTEGER, stoObj.get().trunc());
- } else {
- return new StoreValue(Types.REAL, stoObj.get());
- }
- }
- private store: Map<String, StoreObject>;
- public nextStore?: Store
- public mode: Symbol;
- constructor(public name: String) {
- this.store = new Map<String, StoreObject>();
- this.mode = Modes.RUN;
- }
- extendStore (nextStore: Store) {
- this.nextStore = nextStore;
- }
- applyStore (id: String): IStoreValue {
- if (!this.store.has(id)) {
- if (this.nextStore != null) {
- return this.nextStore.applyStore(id);
- } else {
- throw new Error(`Variable ${id} not found.`);
- }
- }
- const val = this.store.get(id)!;
- let result = null
- if (val.type instanceof ArrayType) {
- const array = val as StoreObjectArray;
- const array_type = array.type as ArrayType;
- let l = 0, c = 0;
- const values = array.value.map( v => {
- if(array.isVector) {
- return new StoreValueAddress(array_type.innerType, v, l++, undefined, array.id, array.readOnly);
- } else {
- if(c >= array.columns) {
- c = 0;
- l += 1;
- }
- return new StoreValueAddress(array_type.innerType, v, l, c++, array.id, array.readOnly);
- }
- });
- result = new ArrayStoreValue(array_type, values, array.lines, array.columns, val.id, val.readOnly);
- } else {
- result = new StoreValue(val.type, val.value, val.id, val.readOnly);
- }
-
- return result;
- }
- updateStore (id: String, stoValue: IStoreValue): Store {
- if (!this.store.has(id)) {
- if (this.nextStore != null) {
- this.nextStore.updateStore(id, stoValue);
- return this;
- } else {
- // TODO: better error message
- throw new Error(`Variable ${id} not found.`);
- }
- } else {
- const oldObj = this.store.get(id)!;
- if (oldObj.readOnly) {
- // TODO: better error message
- throw new Error("Cannot change value of a read only variable: " + id);
- }
- if (oldObj.type instanceof ArrayType) {
- // oldObj.updateRef(stoValue);
- return this;
- } else if (oldObj.isCompatible(stoValue)) {
- // TODO check for array....
- // const loc_address = Location.allocate(stoValue.get());
- // const newObj = new StoreObject(stoValue.type, loc_address, stoValue.isConst);
- // newObj.setID(id);
- // this.store.get(id)!.destroy();
- // this.store.set(id, newObj);
- const loc_address = oldObj.locAddress;
- Location.updateAddress(loc_address, stoValue.get());
- return this;
- } else {
- const oldType = oldObj.type;
- const stoType = stoValue.type;
- // TODO: better error message
- throw new Error(`${oldType} is not compatible with type ${stoType} given`);
- }
- }
- }
- //In case of future use of ref, it needs to have a special function to update the storeRefObject
- // and no the StoreObject refferenced by it
- // updateStoreRef(id, stoObjAddress) {...}
- insertStore (id: String, stoValue: IStoreValue) {
- if (this.store.has(id)) {
- // TODO: better error message
- throw new Error(`${id} is already defined`);
- }
- // TODO check for array....
- let newObj:StoreObject;
- if(stoValue instanceof StoreValueRef) {
- newObj = new StoreObjectRef(stoValue);
- } else if (stoValue instanceof ArrayStoreValueRef) {
- newObj = new StoreObjectArrayRef(stoValue);
- } else if (stoValue instanceof ArrayStoreValue) {
- const columns = stoValue.isVector() ? 0 : stoValue.columns!;
- const addresses: number[] = [];
- const all_values = stoValue.get();
- if(all_values.length > 0) {
- for(let i = 0; i < stoValue.get().length; i += 1) {
- const val = all_values[i].get();
- addresses.push(Location.allocate(val));
- }
- } else {
- let total = stoValue.lines;
- total = stoValue.isVector() ? total : total * columns;
- for(let i = 0; i < total; i += 1) {
- addresses.push(Location.allocate(null));
- }
- }
- newObj = new StoreObjectArray(stoValue.type as ArrayType, stoValue.lines, columns, addresses, stoValue.isConst);
- } else {
- const loc_address = Location.allocate(stoValue.get());
- newObj = new StoreObject(stoValue.type, loc_address, stoValue.isConst);
- }
- newObj.setID(id);
- this.store.set(id, newObj);
- return this;
- }
- /**
- * Helper function similar to applyStore. But it returns the actual object in the store be it ref or not
- * applyStore will return the refferenced object if the object in the store is a ref
- */
- getStoreObject (id: String): StoreObject {
- if (!this.store.has(id)) {
- if (this.nextStore != null) {
- return this.nextStore.getStoreObject(id);
- } else {
- throw new Error(`Variable ${id} not found.`);
- }
- }
- return this.store.get(id)!;
- }
- destroy (): void {
- this.store.forEach(sto => sto.destroy(), this);
- }
- }
|