tock-mirror/UsageCheck.hs
Adam Sampson c8d5a4ff40 Add licensing information.
The compiler itself is under the GPLv2+; the support code that gets built into
user programs is under the LGPLv2+. This matches the existing practice for the
KRoC project.

(As with Occade, I've used the new GPLv3-style license header in the source
files, though, since that avoids having to update the FSF's postal address.)
2007-08-18 20:42:11 +00:00

164 lines
7.8 KiB
Haskell

{-
Tock: a compiler for parallel languages
Copyright (C) 2007 University of Kent
This program is free software; you can redistribute it and/or modify it
under the terms of the GNU General Public License as published by the
Free Software Foundation, either version 2 of the License, or (at your
option) any later version.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License along
with this program. If not, see <http://www.gnu.org/licenses/>.
-}
module UsageCheck where
import qualified AST as A
import Data.Generics
import Metadata
import Data.List
import Data.Maybe
--An obvious thing to do would be to hold these lists (of written/read variables respectively) instead as sets
--However, this would involve defining an ordering over A.Variable. This would be do-able for plain A.Variables,
--but in order to define a proper ordering for subscripted variables, we would end up needing to provide an
--ordering for A.Expression (and all its subtypes)! Therefore, they are kept simply as lists:
type WrittenRead = ([A.Variable],[A.Variable])
concatWR :: WrittenRead -> WrittenRead -> WrittenRead
concatWR (w0,r0) (w1,r1) = (w0 ++ w1,r0 ++ r1)
foldWR :: [WrittenRead] -> WrittenRead
foldWR = foldl1 concatWR
removeDupWR :: WrittenRead -> WrittenRead
removeDupWR (w,r) = (nub w,nub r)
--Gets the (written,read) variables of a piece of an occam program:
--For subscripted variables used as Lvalues , e.g. a[b] it should return a[b] as written-to and b as read
--For subscripted variables used as expressions, e.g. a[b] it should return a[b],b as read (with no written-to)
getVars :: Data t => t -> WrittenRead
getVars
= removeDupWR . (everything concatWR (([],[])
`mkQ` getVarProc
))
where
getVarProc :: A.Process -> WrittenRead
getVarProc (A.Assign _ vars expList)
--Join together:
= concatWR
--The written-to variables on the LHS:
(foldWR (map processVarW vars))
--All variables read on the RHS:
(everything concatWR (([],[]) `mkQ` getVarExp) expList)
--TODO output input etc (all other processes that directly write to/read from variables)
getVarProc _ = ([],[])
{-
Near the beginning, this piece of code was too clever for itself and applied processVarW using "everything".
The problem with this is that given var@(A.SubscriptedVariable _ sub arrVar), the functions would be recursively
applied to sub and arrVar. processVarW should return var, but never the subscripts in sub; those subscripts are not written to!
Therefore processVarW must *not* be applied using the generics library, and instead should always be applied
directly to an A.Variable. Internally it uses the generics library to process the subscripts (using getVarExp)
-}
--Pull out all the subscripts into the read category, but leave the given var in the written category:
processVarW :: A.Variable -> WrittenRead
processVarW v@(A.Variable _ _) = ([v],[])
processVarW v@(A.SubscriptedVariable _ s _) = concatWR ([v],[]) (everything concatWR (([],[]) `mkQ` getVarExp) s)
--Only need to deal with the two cases where we can see an A.Variable directly;
--the generic recursion will take care of nested expressions, and even the expressions used as subscripts
getVarExp :: A.Expression -> WrittenRead
getVarExp (A.SizeVariable _ v) = ([],[v])
getVarExp (A.ExprVariable _ v) = ([],[v])
getVarExp _ = ([],[])
-- I am not sure how you could build this out of the standard functions, so I built it myself
--Takes a list (let's say Y), a function that applies to a single item and a list, and then goes through applying the function
--to each item in the list, with the rest of the list Y as a parameter. Perhaps the code is clearer:
permuteHelper :: (a -> [a] -> b) -> [a] -> [b]
permuteHelper _ [] = []
permuteHelper func (x:xs) = permuteHelper' func [] x xs
where
permuteHelper' :: (a -> [a] -> b) -> [a] -> a -> [a] -> [b]
permuteHelper' func prev cur [] = [func cur prev]
permuteHelper' func prev cur (next:rest) = (func cur (prev ++ (next:rest))) : (permuteHelper' func (prev ++ [cur]) next rest)
--Whereas the other passes (at the current time of writing) are transforms on the tree, the usage checker
--does not modify the tree at all; it only needs to check if the usage is valid or not. Therefore instead
--of using the generic "everywhere" function with a transform, I use "listify" (which is built on top of "everything")
--to pick out the processes that are failing the check
--Returns Nothing if the check is fine, or Just [A.Process] if there is an error (listing all processes that are in error)
parUsageCheck :: A.Process -> Maybe [A.Process]
parUsageCheck proc
= case listify doUsageCheck proc of
[] -> Nothing
x -> Just x
where
doUsageCheck :: A.Process -> Bool
doUsageCheck (A.Par _ _ s)
--Looking at the AST Parse for occam, we can either have:
--A.Par _ _ (A.Several _ [A.OnlyP _ _])
--A.Par _ _ (A.Rep _ _ (A.OnlyP _ _))
--Therefore skipSpecs shouldn't be necessary, but I may as well keep it in for now:
= case skipSpecs s of
A.Several _ structList ->
--Need to check that for each written item, it is not read/written elsewhere:
or $ permuteHelper usageCheckList (map getVars structList)
A.Rep _ rep (A.OnlyP _ proc) ->
False --TODO!
doUsageCheck _ = False
--Recursively skips down past the Specs:
skipSpecs :: A.Structured -> A.Structured
skipSpecs (A.Spec _ _ s) = skipSpecs s
skipSpecs other = other
--We need to check:
--a) Should be no intersection between our written items, and any written or read items anywhere else
--b) You may only use a variable subscript if the array is not used anywhere else in the PAR
--We don't actually need to check for constant subscripts being the same - we assume constant folding has already
--taken place, which means that a[0] and a[0] will be picked up by the first check (a). This also assumes
--that all array index literals will have been converted into INT literals rather than any other type.
--The occam 2 manual says the types must be type INT, so this seems like an okay assumption.
usageCheckList :: WrittenRead -> [WrittenRead] -> Bool
usageCheckList (written,read) others
= ((length (intersect written (allOtherWritten ++ allOtherRead))) /= 0)
|| ((length (intersect (varSubscriptedArrays written) (subscriptedArrays (allOtherWritten ++ allOtherRead)))) /= 0)
where
allOtherWritten = concatMap fst others
allOtherRead = concatMap snd others
--Takes in the subscripted compound variables, and returns the *array* variables (not the subscripted compounds)
varSubscriptedArrays :: [A.Variable] -> [A.Variable]
varSubscriptedArrays = mapMaybe varSubscriptedArrays'
varSubscriptedArrays' :: A.Variable -> Maybe A.Variable
varSubscriptedArrays' (A.SubscriptedVariable _ s v)
= case ((length . snd . removeDupWR) (everything concatWR (([],[]) `mkQ` getVarExp) s)) of
0 -> Nothing
_ -> Just v
varSubscriptedArrays' _ = Nothing
--Takes in the subscripted compound variables, and returns the *array* variables (not the subscripted compounds)
subscriptedArrays :: [A.Variable] -> [A.Variable]
subscriptedArrays = mapMaybe subscriptedArrays'
subscriptedArrays' :: A.Variable -> Maybe A.Variable
subscriptedArrays' (A.SubscriptedVariable _ _ v) = Just v
subscriptedArrays' _ = Nothing