module Camfort.Specification.Stencils.Synthesis where
import Data.Data
import Data.List
import Data.Maybe
import qualified Data.Map as M
import Data.Generics.Uniplate.Operations
import Control.Monad.State.Lazy
import Control.Monad.Reader
import Control.Monad.Writer hiding (Product)
import Camfort.Specification.Stencils.InferenceBackend
import Camfort.Specification.Stencils.Syntax
import Camfort.Specification.Stencils.Model
import Camfort.Analysis.Loops (collect)
import Camfort.Analysis.Annotations
import Camfort.Helpers.Vec
import Camfort.Helpers hiding (lineCol, spanLineCol)
import qualified Language.Fortran.AST as F
import qualified Language.Fortran.Analysis as FA
import qualified Language.Fortran.Analysis.Types as FAT
import qualified Language.Fortran.Analysis.Renaming as FAR
import qualified Language.Fortran.Analysis.BBlocks as FAB
import qualified Language.Fortran.Analysis.DataFlow as FAD
import qualified Language.Fortran.Util.Position as FU
import Language.Fortran.Util.Position
import Data.Map hiding (map)
formatSpec ::
Maybe String
-> FAR.NameMap
-> (FU.SrcSpan, Either [([Variable], Specification)] (String,Variable))
-> String
formatSpec prefix nm (span, Right (evalInfo,name)) =
prefix'
++ evalInfo
++ (if name /= "" then " :: " ++ realName name else "") ++ "\n"
where
realName v = v `fromMaybe` (v `M.lookup` nm)
prefix' = case prefix of
Nothing -> show (spanLineCol span) ++ " \t"
Just pr -> pr
formatSpec prefix nm (span, Left []) = ""
formatSpec prefix nm (span, Left specs) =
(intercalate "\n" $ map (\s -> prefix' ++ doSpec s) specs)
where
prefix' = case prefix of
Nothing -> show (spanLineCol span) ++ " \t"
Just pr -> pr
commaSep = intercalate ", "
doSpec (arrayVar, spec) =
show (fixSpec spec) ++ " :: " ++ commaSep (map realName arrayVar)
realName v = v `fromMaybe` (v `M.lookup` nm)
fixSpec (Specification (Right (Dependency vs b))) =
Specification (Right (Dependency (map realName vs) b))
fixSpec s = s
lineCol :: FU.Position -> (Int, Int)
lineCol p = (fromIntegral $ FU.posLine p, fromIntegral $ FU.posColumn p)
spanLineCol :: FU.SrcSpan -> ((Int, Int), (Int, Int))
spanLineCol (FU.SrcSpan l u) = (lineCol l, lineCol u)
a = (head $ FA.initAnalysis [unitAnnotation]) { FA.insLabel = Just 0 }
s = SrcSpan (Position 0 0 0) (Position 0 0 0)
synthesise :: Specification -> F.Name -> [F.Name] -> [F.Expression (FA.Analysis A)]
synthesise (Specification (Left (Exact spec))) v ixs =
map toSubscriptExpr . toList . fromExact . model $ (Exact spec)
where toSubscriptExpr (offs,_) = ixExprToSubscript v
. map (uncurry offsetToIx) $ zip ixs offs
synthesise _ _ _ = []
ixExprToSubscript :: F.Name -> [F.Index (FA.Analysis A)] -> F.Expression (FA.Analysis A)
ixExprToSubscript v es =
F.ExpSubscript a s (F.ExpValue a s (F.ValVariable v)) (F.AList a s es)
offsetToIx :: F.Name -> Int -> F.Index (FA.Analysis A)
offsetToIx v o
| o == absoluteRep
= F.IxSingle a s Nothing (F.ExpValue a s (F.ValInteger "0"))
| o == 0 = F.IxSingle a s Nothing (F.ExpValue a s (F.ValVariable v))
| o > 0 = F.IxSingle a s Nothing (F.ExpBinary a s F.Addition
(F.ExpValue a s (F.ValVariable v))
(F.ExpValue a s (F.ValInteger $ show o)))
| otherwise = F.IxSingle a s Nothing (F.ExpBinary a s F.Subtraction
(F.ExpValue a s (F.ValVariable v))
(F.ExpValue a s (F.ValInteger $ show (abs o))))
offsetToIxWithIVs :: [Variable] -> F.Name -> Int -> F.Index (FA.Analysis A)
offsetToIxWithIVs ivs v o = F.setAnnotation a' ix
where a' = a { FA.prevAnnotation = (FA.prevAnnotation a) {indices = ivs} }
ix = offsetToIx v o