From 8fcbd2aa73cb380349b88c73a58319fc50ced88a Mon Sep 17 00:00:00 2001 From: koen <koenwermer@gmail.com> Date: Fri, 9 Dec 2016 23:39:40 +0100 Subject: [PATCH] Method calls implemented. Constructor methods and method arguments not yet implemented. --- HelperFunctions.hs | 9 ++++++++ Substitute.hs | 18 ++++++++-------- Tests/methods.java | 15 +++++++++---- WLP.hs | 53 ++++++++++++++++++++++++++-------------------- 4 files changed, 59 insertions(+), 36 deletions(-) diff --git a/HelperFunctions.hs b/HelperFunctions.hs index 6512372..1b0c1c4 100644 --- a/HelperFunctions.hs +++ b/HelperFunctions.hs @@ -10,6 +10,7 @@ type CallCount = [(Ident, Int)] -- | Retrieves the type from the environment lookupType :: [TypeDecl] -> TypeEnv -> Name -> Type +lookupType decls env (Name [(Ident s@('$':_))]) = getReturnVarType decls s -- Names starting with a '$' symbol are generated and represent the return variable of a function lookupType decls env (Name idents) = case lookup (Name [head idents]) env of Just t -> getFieldType decls t (Name (tail idents)) Nothing -> error ("can't find type of " ++ prettyPrint (Name idents) ++ "\r\n TypeEnv: " ++ show env) @@ -34,6 +35,14 @@ getFieldType decls (RefType (ClassRefType t)) (Name (f:fs)) = getFieldType decls _ -> getDecl t xs getDecl _ _ = error "nested class" +-- | Creates a string that that represents the return var name of a method call. This name is extended by a number to indicate the depth of the recursive calls +makeReturnVarName :: Ident -> String +makeReturnVarName (Ident s) = "$" ++ s ++ "$" + +-- | Get's the type of a generated variable +getReturnVarType :: [TypeDecl] -> String -> Type +getReturnVarType decls s = fromJust $ getMethodType decls (Ident (takeWhile (/= '$') (tail s))) + -- Increments the call count for a given method incrCallCount :: CallCount -> Ident -> CallCount incrCallCount [] id = [(id, 1)] diff --git a/Substitute.hs b/Substitute.hs index 41e5bdf..ed79237 100644 --- a/Substitute.hs +++ b/Substitute.hs @@ -47,8 +47,8 @@ substVarExpAlgebra = (fLit, fClassLit, fThis, fThisClass, fInstanceCreation, fQu _ -> ExpName (Name name) -- done: Just [] -> ExpName (Name name) - -- check a combination and recurse: - Just ((nameInit, lhsNameInit) : combs') -> flip (foldExp substVarExpAlgebra) (inh {combs = Just combs'}) $ + -- check a combination and recurse when necessary: + Just ((nameInit, lhsNameInit) : combs') -> case lhs inh of NameLhs (Name lhsName) -> case lookupType (decls inh) (env inh) (Name lhsNameInit) of PrimType _ | lhsName == name -> rhs inh @@ -56,24 +56,24 @@ substVarExpAlgebra = (fLit, fClassLit, fThis, fThisClass, fInstanceCreation, fQu ExpName (Name rhsName) | take (length lhsName) name == lhsName -> ExpName (Name (rhsName ++ drop (length lhsName) name)) -- accessing o1.x might affect o2.x if o1 and o2 point to the same object: | name == nameInit ++ drop (length lhsNameInit) lhsName && length name > length nameInit -> Cond (ExpName (Name nameInit) ==* ExpName (Name lhsNameInit)) (rhs inh) (ExpName (Name name)) - | otherwise -> ExpName (Name name) + | otherwise -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) -- we substitute instance creation only if we access a field, to not lose pointer information -- example: {x = new obj} doesn't affect {x = y} but it does affect {x.a = y.a} InstanceCreation _ _ _ _ | length lhsName < length name && take (length lhsName) name == lhsName -> getFields (decls inh) (rhs inh) (drop (length lhsName) name) | length lhsName < length name && take (length lhsName) name == nameInit ++ drop (length nameInit) lhsName -> Cond (ExpName (Name nameInit) ==* ExpName (Name lhsNameInit)) (getFields (decls inh) (rhs inh) (drop (length lhsName) name)) (ExpName (Name name)) - | otherwise -> ExpName (Name name) + | otherwise -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) -- the same idea for arrays - ArrayCreate _ _ _ | not $ arrayLookup inh -> ExpName (Name name) - ArrayCreateInit _ _ _ | not $ arrayLookup inh -> ExpName (Name name) + ArrayCreate _ _ _ | not $ arrayLookup inh -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) + ArrayCreateInit _ _ _ | not $ arrayLookup inh -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) _ | take (length lhsName) name == lhsName -> getFields (decls inh) (rhs inh) (drop (length lhsName) name) | name == nameInit ++ drop (length lhsNameInit) lhsName -> Cond (ExpName (Name nameInit) ==* ExpName (Name lhsNameInit)) (rhs inh) (ExpName (Name name)) -- the assignment doesn't affect this expression: - | otherwise -> ExpName (Name name) - _ -> ExpName (Name name) - _ -> ExpName (Name name) + | otherwise -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) + _ -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) + _ -> foldExp substVarExpAlgebra (ExpName (Name name)) (inh {combs = Just combs'}) fPostIncrement e inh = PostIncrement (e inh) fPostDecrement e inh = PostDecrement (e inh) fPreIncrement e inh = PreIncrement (e inh) diff --git a/Tests/methods.java b/Tests/methods.java index 1fd298d..0ded15c 100644 --- a/Tests/methods.java +++ b/Tests/methods.java @@ -1,17 +1,19 @@ public static class Main { - static int x; + static int x, y; public static void main(String[] args) { C1 c1 = new C1(); - c1.c += 2; //c1.method1(); + c1.method1(); + c1.method1(); x = c1.c; } - public static void method() + public static int method() { + return 1; } } @@ -25,6 +27,11 @@ public class C1 public void method1() { - // c++; + this.c += 1; + } + + public int method2() + { + return 1; } } \ No newline at end of file diff --git a/WLP.hs b/WLP.hs index 675a171..eaaf3c1 100644 --- a/WLP.hs +++ b/WLP.hs @@ -144,22 +144,22 @@ wlpExpAlgebra :: ExpAlgebra (Inh -> (Exp, Syn)) wlpExpAlgebra = (fLit, fClassLit, fThis, fThisClass, fInstanceCreation, fQualInstanceCreation, fArrayCreate, fArrayCreateInit, fFieldAccess, fMethodInv, fArrayAccess, fExpName, fPostIncrement, fPostDecrement, fPreIncrement, fPreDecrement, fPrePlus, fPreMinus, fPreBitCompl, fPreNot, fCast, fBinOp, fInstanceOf, fCond, fAssign, fLambda, fMethodRef) where fLit lit inh = (Lit lit, (acc inh, env inh)) fClassLit mType inh = (ClassLit mType, (acc inh, env inh)) - fThis inh = (This, (acc inh, env inh)) + fThis inh = (fromJust (object inh), (acc inh, env inh)) fThisClass name inh = (ThisClass name, (acc inh, env inh)) fInstanceCreation typeArgs t args mBody inh = (InstanceCreation typeArgs t args mBody, (acc inh, env inh)) fQualInstanceCreation e typeArgs t args mBody inh = (QualInstanceCreation (getExp e inh) typeArgs t args mBody, (getTrans e inh, env inh)) fArrayCreate t dimLengths dim inh = (ArrayCreate t (map (flip getExp inh) dimLengths) dim, (acc inh, env inh)) fArrayCreateInit t dim init inh = (ArrayCreateInit t dim init, (acc inh, env inh)) - fFieldAccess fieldAccess inh = (FieldAccess fieldAccess, (acc inh, env inh)) + fFieldAccess fieldAccess inh = (ExpName (foldFieldAccess inh fieldAccess), (acc inh, env inh)) fMethodInv invocation inh = (ExpName (Name [getReturnVar inh invocation]), (if getCallCount (calls inh) (invocationToId invocation) >= nrOfUnroll then const true - else fst (wlp' (inh {env = maybe (env inh) (\t -> (Name [getReturnVar inh invocation], t) : env inh) (getType inh invocation), calls = incrCallCount (calls inh) (invocationToId invocation), ret = Just (getReturnVar inh invocation), object = Just (getObject invocation)}) (inlineMethod inh invocation)) . acc inh, env inh)) + else fst (wlp' (inh {acc = id, calls = incrCallCount (calls inh) (invocationToId invocation), ret = Just (getReturnVar inh invocation), object = getObject invocation}) (inlineMethod inh invocation)) . acc inh, env inh)) fArrayAccess (ArrayIndex a i) inh = case catch inh of Nothing -> (arrayAccess a i, (acc inh, env inh)) Just (cs, f) -> (arrayAccess a i, (arrayAccessWlp a i inh, env inh)) - fExpName name inh = (varInObject (object inh) name, (acc inh, env inh)) + fExpName name inh = (ExpName name, (acc inh, env inh)) -- x++ increments x but evaluates to the original value fPostIncrement e inh = case getExp e inh of var@(ExpName name) -> (var, (substVar (env inh) (decls inh) (NameLhs name) (BinOp var Add (Lit (Int 1))) . acc inh, env inh)) @@ -179,12 +179,12 @@ wlpExpAlgebra = (fLit, fClassLit, fThis, fThisClass, fInstanceCreation, fQualIns fPreBitCompl e inh = (PreBitCompl $ getExp e inh, (acc inh, env inh)) fPreNot e inh = (PreNot $ getExp e inh, (acc inh, env inh)) fCast t e inh = (getExp e inh, (acc inh, env inh)) - fBinOp e1 op e2 inh = (BinOp (getExp e1 inh) op (getExp e2 inh), (getTrans e1 (inh {acc = getTrans e2 inh}), env inh)) + fBinOp e1 op e2 inh = (BinOp (getExp e1 inh) op (getExp e2 inh), (getTrans e2 (inh {acc = getTrans e1 inh}), env inh)) fInstanceOf = error "instanceOf" fCond g e1 e2 inh = (Cond (getExp g inh) (getExp e1 inh) (getExp e2 inh), (getTrans g (inh {acc = id}) . (\q -> (getExp g inh &* getTrans e1 (inh {acc = id}) q) |* (neg (getExp g inh) &* getTrans e2 (inh {acc = id}) q)) . acc inh, env inh)) - fAssign lhs op e inh = let lhs' = lhsInObject (object inh) lhs + fAssign lhs op e inh = let lhs' = foldLhs inh lhs rhs' = desugarAssign lhs' op (getExp e inh) - in (rhs', (substVar (env inh) (decls inh) lhs' rhs' . getTrans e inh, env inh)) + in (rhs', (getTrans e inh {acc = id} . substVar (env inh) (decls inh) lhs' rhs' . acc inh, env inh)) fLambda = error "lambda" fMethodRef = error "method reference" @@ -207,13 +207,14 @@ wlpExpAlgebra = (fLit, fClassLit, fThis, fThisClass, fInstanceCreation, fQualIns -- Gets the variable that represents the return value of the method getReturnVar :: Inh -> MethodInvocation -> Ident - getReturnVar inh invocation = Ident (show (invocationToId invocation) ++ show (getCallCount (calls inh) (invocationToId invocation))) + getReturnVar inh invocation = Ident (makeReturnVarName (invocationToId invocation) ++ show (getCallCount (calls inh) (invocationToId invocation))) -- Gets the object a method is called from - getObject :: MethodInvocation -> Exp - getObject (MethodCall name _) = ExpName (Name (take (length (fromName name) - 1) (fromName name))) - getObject (PrimaryMethodCall e _ _ _) = e - getObject _ = undefined + getObject :: MethodInvocation -> Maybe Exp + getObject (MethodCall name _) | length (fromName name) > 1 = Just (ExpName (Name (take (length (fromName name) - 1) (fromName name)))) + | otherwise = Nothing + getObject (PrimaryMethodCall e _ _ _) = Just e + getObject _ = undefined -- Gets the return type of a method getType :: Inh -> MethodInvocation -> Maybe Type @@ -225,17 +226,23 @@ wlpExpAlgebra = (fLit, fClassLit, fThis, fThisClass, fInstanceCreation, fQualIns invocationToId (PrimaryMethodCall _ _ id _) = id invocationToId _ = undefined - -- Changes the lhs to refer to a field of a given object - lhsInObject :: Maybe Exp -> Lhs -> Lhs - lhsInObject Nothing lhs = lhs - lhsInObject (Just obj) lhs = case lhs of - NameLhs name -> FieldLhs (fieldAccess obj name) - _ -> error "TODO: lhsInObject" - - -- Changes the var to refer to a field of a given object - varInObject :: Maybe Exp -> Name -> Exp - varInObject Nothing name = ExpName name - varInObject (Just obj) name = FieldAccess (fieldAccess obj name) + -- Folds the expression part of an lhs + foldLhs :: Inh -> Lhs -> Lhs + foldLhs inh lhs = case lhs of + FieldLhs (PrimaryFieldAccess e id) -> case getExp (foldExp wlpExpAlgebra e) inh of + ExpName name -> NameLhs (Name (fromName name ++ [id])) + _ -> error "foldFieldAccess" + ArrayLhs (ArrayIndex e i) -> ArrayLhs (ArrayIndex (getExp (foldExp wlpExpAlgebra e) inh) (map (\e -> getExp (foldExp wlpExpAlgebra e) inh) i)) + lhs' -> lhs' + + -- Folds the expression part of a fieldaccess and simplifies it to a name + foldFieldAccess :: Inh -> FieldAccess -> Name + foldFieldAccess inh fieldAccess = case fieldAccess of + PrimaryFieldAccess e id -> case getExp (foldExp wlpExpAlgebra e) inh of + ExpName name -> Name (fromName name ++ [id]) + _ -> error "foldFieldAccess" + SuperFieldAccess id -> foldFieldAccess inh (PrimaryFieldAccess (fromJust (object inh)) id) + ClassFieldAccess name id -> Name (fromName name ++ [id]) -- | Gets the expression attribute getExp :: (Inh -> (Exp, Syn)) -> Inh -> Exp -- GitLab