-- -- Copyright (c) 2009-2010, ERICSSON AB All rights reserved. -- -- Redistribution and use in source and binary forms, with or without -- modification, are permitted provided that the following conditions are met: -- -- * Redistributions of source code must retain the above copyright notice, -- this list of conditions and the following disclaimer. -- * Redistributions in binary form must reproduce the above copyright -- notice, this list of conditions and the following disclaimer in the -- documentation and/or other materials provided with the distribution. -- * Neither the name of the ERICSSON AB nor the names of its contributors -- may be used to endorse or promote products derived from this software -- without specific prior written permission. -- -- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" -- AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE -- IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE -- ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS -- BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, -- OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF -- SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS -- INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN -- CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) -- ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF -- THE POSSIBILITY OF SUCH DAMAGE. -- module Feldspar.Compiler.Standalone.Options where import qualified Feldspar.Compiler.Options as CompilerCoreOptions import qualified Feldspar.Compiler.Compiler as CompilerCore import qualified Feldspar.Compiler.Standalone.Library as StandaloneLib import Feldspar.Compiler.Standalone.Constants import Feldspar.Compiler.Platforms import Data.List import Data.Char import System.Console.GetOpt import System.Exit import System.Environment import System.IO import System.Process import System.Info import System.Directory availablePlatformsStrRep = StandaloneLib.formatStringList $ map (StandaloneLib.upperFirst . CompilerCoreOptions.name) availablePlatforms data FunctionMode = SingleFunction String | MultiFunction data Options = Options { optSingleFunction :: FunctionMode , optOutputFileName :: Maybe String , optDotGeneration :: Bool , optDotFileName :: Maybe String , optCompilerMode :: CompilerCoreOptions.Options } -- | Default options startOptions :: Options startOptions = Options { optSingleFunction = MultiFunction , optOutputFileName = Nothing , optDotGeneration = False , optDotFileName = Nothing , optCompilerMode = CompilerCore.defaultOptions } -- | Option descriptions for getOpt optionDescriptors :: [ OptDescr (Options -> IO Options) ] optionDescriptors = [ Option "f" ["singlefunction"] (ReqArg (\arg opt -> return opt { optSingleFunction = SingleFunction arg }) "FUNCTION") "Enables single-function compilation" , Option "o" ["output"] (ReqArg (\arg opt -> return opt { optOutputFileName = Just arg }) "outputfile.c") "Overrides the file name for the generated output code" , Option "d" ["todot"] (OptArg (\arg opt -> return opt { optDotFileName = arg, optDotGeneration = True }) "dotfile.dot") "Enables dot generation (outputs to stdout if no filename is specified)" , Option "p" ["platform"] (ReqArg (\arg opt -> return opt { optCompilerMode = (optCompilerMode opt) { CompilerCoreOptions.platform = decodePlatform arg } }) "") ("Overrides the target platform " ++ availablePlatformsStrRep) , Option "u" ["unroll"] (ReqArg (\arg opt -> return opt { optCompilerMode = (optCompilerMode opt) { CompilerCoreOptions.unroll = CompilerCoreOptions.Unroll (parseInt arg "Invalid unroll count") } }) "") "Enables loop unrolling" , Option "D" ["debuglevel"] (ReqArg (\arg opt -> return opt { optCompilerMode = (optCompilerMode opt) { CompilerCoreOptions.debug = decodeDebug arg } }) "") "Specifies debug level (NoSimplification | NoPrimitiveInstructionHandling)" , Option "a" ["defaultArraySize"] (ReqArg (\arg opt -> return opt { optCompilerMode = (optCompilerMode opt) { CompilerCoreOptions.defaultArraySize = parseInt arg "Invalid default array size" } }) "") "Overrides default array size" , Option "h" ["help"] (NoArg (\_ -> do --prg <- getProgName hPutStrLn stderr (usageInfo helpHeader optionDescriptors) exitWith ExitSuccess)) "Show this help message" ] -- ============================================================================== -- == Option Decoders -- ============================================================================== findPlatformByName :: String -> Maybe CompilerCoreOptions.Platform findPlatformByName platformName = -- Finds a platform by name using case-insensitive comparison find (\platform -> (map toLower platformName) == (map toLower $ CompilerCoreOptions.name platform)) availablePlatforms decodePlatform :: String -> CompilerCoreOptions.Platform decodePlatform s = case (findPlatformByName s) of Just platform -> platform Nothing -> error $ "Invalid platform specified. Valid platforms are: " ++ availablePlatformsStrRep decodeDebug "NoSimplification" = CompilerCoreOptions.NoSimplification decodeDebug "NoPrimitiveInstructionHandling" = CompilerCoreOptions.NoPrimitiveInstructionHandling decodeDebug _ = error "Invalid debug level specified" parseInt :: String -> String -> Int parseInt arg message = case reads arg of [(x, "")] -> x _ -> error message