module Camfort.Specification.Units
(checkUnits, inferUnits, synthesiseUnits, inferCriticalVariables)
where
import qualified Data.Map.Strict as M
import Data.Char (isNumber)
import Data.List (intercalate)
import Data.Maybe (fromMaybe, maybeToList, listToMaybe, mapMaybe)
import Data.Generics.Uniplate.Operations
import Control.Monad.State.Strict
import Camfort.Helpers hiding (lineCol)
import Camfort.Helpers.Syntax
import Camfort.Output
import Camfort.Analysis.Annotations
import Camfort.Input
import Camfort.Specification.Units.Environment
import Camfort.Specification.Units.Monad
import Camfort.Specification.Units.InferenceBackend
import Camfort.Specification.Units.InferenceFrontend
import Camfort.Specification.Units.Synthesis (runSynthesis)
import qualified Language.Fortran.Analysis.Renaming as FAR
import qualified Language.Fortran.Analysis as FA
import qualified Language.Fortran.AST as F
import qualified Language.Fortran.Util.Position as FU
import qualified Debug.Trace as D
inferCriticalVariables
:: UnitOpts -> (Filename, F.ProgramFile Annotation) -> (Report, Int)
inferCriticalVariables uo (fname, pf)
| Right vars <- eVars = okReport vars
| Left exc <- eVars = (errReport exc, 1)
where
okReport [] = (logs ++ "\n" ++ fname
++ ":No additional annotations are necessary.\n", 0)
okReport vars = (logs ++ "\n" ++ fname ++ ": "
++ show numVars
++ " variable declarations suggested to be given a specification:\n"
++ unlines [ " " ++ expReport ei | ei <- expInfo ], numVars)
where
names = map showVar vars
expInfo = filter ((`elem` names) . FA.srcName) $ declVariables pfUA
numVars = length expInfo
expReport e = "(" ++ showSrcSpan (FU.getSpan e) ++ ") " ++ FA.srcName e
varReport = intercalate ", " . map showVar
showVar (UnitVar (_, s)) = s
showVar (UnitLiteral _) = "<literal>"
showVar _ = "<bad>"
errReport exc = logs ++ "\n" ++ fname ++ ":\n" ++ show exc
uOpts = uo { uoNameMap = nameMap }
(eVars, state, logs) = runUnitSolver uOpts pfRenamed $ initInference >> runCriticalVariables
pfUA = usProgramFile state
pfRenamed = FAR.analyseRenames . FA.initAnalysis . fmap mkUnitAnnotation $ pf
nameMap = FAR.extractNameMap pfRenamed
checkUnits, inferUnits
:: UnitOpts -> (Filename, F.ProgramFile Annotation) -> Report
checkUnits uo (fname, pf)
| Right mCons <- eCons = okReport mCons
| Left exc <- eCons = errReport exc
where
okReport Nothing = logs ++ "\n" ++ fname ++ ": Consistent. " ++ show nVars ++ " variables checked."
okReport (Just cons) = logs ++ "\n" ++ fname ++ ": Inconsistent:\n" ++ reportErrors cons
reportErrors cons = unlines [ reportError con | con <- cons ]
reportError con = " - at " ++ srcSpan con
++ pprintConstr (orient (unrename nameMap con))
++ additionalInfo con
where
additionalInfo con =
if null (errorInfo con)
then ""
else "\n instead" ++ intercalate "\n" (mapNotFirst (pad 10) (errorInfo con))
errorInfo con =
[" '" ++ sName ++ "' is '" ++ pprintUnitInfo (unrename nameMap u) ++ "'"
| UnitVar (vName, sName) <- universeBi con
, u <- findUnitConstrFor con vName ]
findUnitConstrFor con v = mapMaybe (\con' -> if con == con'
then Nothing
else constrainedTo v con')
(concat $ M.elems templateMap)
constrainedTo v (ConEq (UnitVar (v', _)) u) | v == v' = Just u
constrainedTo v (ConEq u (UnitVar (v', _))) | v == v' = Just u
constrainedTo _ _ = Nothing
mapNotFirst f [] = []
mapNotFirst f (x : xs) = x : (map f xs)
orient (ConEq u (UnitVar v)) = ConEq (UnitVar v) u
orient c = c
pad o = (++) (replicate o ' ')
srcSpan con | Just ss <- findCon con = showSrcSpan ss ++ " "
| otherwise = ""
findCon :: Constraint -> Maybe FU.SrcSpan
findCon con = listToMaybe $ [ FU.getSpan x | x <- universeBi pfUA :: [F.Expression UA], getConstraint x `eq` con ] ++
[ FU.getSpan x | x <- universeBi pfUA :: [F.Statement UA] , getConstraint x `eq` con ] ++
[ FU.getSpan x | x <- universeBi pfUA :: [F.Declarator UA], getConstraint x `eq` con ] ++
[ FU.getSpan x | x <- universeBi pfUA :: [F.Argument UA] , getConstraint x `eq` con ]
where eq Nothing _ = False
eq (Just c1) c2 = conParamEq c1 c2
varReport = intercalate ", " . map showVar
showVar (UnitVar (_, s)) = s
showVar (UnitLiteral _) = "<literal>"
showVar _ = "<bad>"
errReport exc = logs ++ "\n" ++ fname ++ ": " ++ show exc
uOpts = uo { uoNameMap = nameMap }
(eCons, state, logs) = runUnitSolver uOpts pfRenamed $ initInference >> runInconsistentConstraints
templateMap = usTemplateMap state
pfUA :: F.ProgramFile UA
pfUA = usProgramFile state
nVars = M.size . M.filter (not . isParametricUnit) $ usVarUnitMap state
isParametricUnit u = case u of UnitParamPosAbs {} -> True; UnitParamPosUse {} -> True
UnitParamVarAbs {} -> True; UnitParamVarUse {} -> True
_ -> False
pfRenamed = FAR.analyseRenames . FA.initAnalysis . fmap mkUnitAnnotation $ pf
nameMap = FAR.extractNameMap pfRenamed
inferUnits uo (fname, pf)
| Right [] <- eVars = checkUnits uo (fname, pf)
| Right vars <- eVars = okReport vars
| Left exc <- eVars = errReport exc
where
okReport vars = logs ++ "\n" ++ fname ++ ":\n" ++ unlines [ expReport ei | ei <- expInfo ]
where
expInfo = [ (e, u) | e <- declVariables pfUA
, u <- maybeToList ((FA.varName e, FA.srcName e) `lookup` vars) ]
expReport (e, u) = " " ++ showSrcSpan (FU.getSpan e) ++ " unit " ++ show u ++ " :: " ++ FA.srcName e
errReport exc = logs ++ "\n" ++ fname ++ ": " ++ show exc
uOpts = uo { uoNameMap = nameMap }
(eVars, state, logs) = runUnitSolver uOpts pfRenamed $ initInference >> runInferVariables
pfUA = usProgramFile state
pfRenamed = FAR.analyseRenames . FA.initAnalysis . fmap mkUnitAnnotation $ pf
nameMap = FAR.extractNameMap pfRenamed
synthesiseUnits :: UnitOpts
-> Char
-> (Filename, F.ProgramFile Annotation)
-> (Report, (Filename, F.ProgramFile Annotation))
synthesiseUnits uo marker (fname, pf)
| Right [] <- eVars = (checkUnits uo (fname, pf), (fname, pf))
| Right vars <- eVars = (okReport vars, (fname, pfFinal))
| Left exc <- eVars = (errReport exc, (fname, pfFinal))
where
okReport vars = logs ++ "\n" ++ fname ++ ":\n" ++ unlines [ expReport ei | ei <- expInfo ]
where
expInfo = [ (e, u) | e <- declVariables pfUA
, u <- maybeToList ((FA.varName e, FA.srcName e) `lookup` vars) ]
expReport (e, u) = " " ++ showSrcSpan (FU.getSpan e) ++ " unit " ++ show u ++ " :: " ++ FA.srcName e
errReport exc = logs ++ "\n" ++ fname ++ ": " ++ show exc
uOpts = uo { uoNameMap = nameMap }
(eVars, state, logs) = runUnitSolver uOpts pfRenamed $ initInference >> runInferVariables >>= runSynthesis marker
pfUA = usProgramFile state
pfFinal = fmap prevAnnotation . fmap FA.prevAnnotation $ pfUA
pfRenamed = FAR.analyseRenames . FA.initAnalysis . fmap mkUnitAnnotation $ pf
nameMap = FAR.extractNameMap pfRenamed
unrename nameMap = transformBi $ \ x -> x `fromMaybe` M.lookup x nameMap
showSrcSpan :: FU.SrcSpan -> String
showSrcSpan (FU.SrcSpan l u) = show l
declVariables :: F.ProgramFile UA -> [F.Expression UA]
declVariables pf = flip mapMaybe (universeBi pf) $ \ d -> case d of
F.DeclVariable _ _ v@(F.ExpValue _ _ (F.ValVariable _)) _ _ -> Just v
F.DeclArray _ _ v@(F.ExpValue _ _ (F.ValVariable _)) _ _ _ -> Just v
_ -> Nothing