-- C->Haskell Compiler: interface to C processing routines -- -- Author : Manuel M. T. Chakravarty -- Created: 12 August 99 -- -- Copyright (c) 1999 Manuel M. T. Chakravarty -- -- This file 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 file 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. -- --- DESCRIPTION --------------------------------------------------------------- -- -- This modules provides access to the C processing routines for the rest of -- the compiler. -- --- DOCU ---------------------------------------------------------------------- -- -- language: Haskell 98 -- --- TODO ---------------------------------------------------------------------- -- -- module C2HS.C (-- interface to KL for all non-KL modules -- -- stuff from `Common' (reexported) -- Pos(posOf), -- -- structure tree -- module Language.C.Syntax, -- -- attributed structure tree with operations (reexported from -- `CAttrs') -- AttrC, CObj(..), CTag(..), CDef(..), lookupDefObjC, lookupDefTagC, getDefOfIdentC, -- -- support for C structure tree traversals -- module C2HS.C.Trav, -- loadAttrC, -- locally defined -- -- misc. reexported stuff -- Ident, NodeInfo, Attr(..), -- -- misc. own stuff -- csuffix, hsuffix, isuffix) where import Language.C.Data import Language.C.Syntax import Language.C.Parser import Data.Attributes (Attr(..)) import qualified Data.ByteString.Char8 as BS import C2HS.State (CST, fatal, errorsPresent, showErrors, raiseError, getNameSupply, setNameSupply, Traces(..), putTraceStr) import System.CIO as CIO (liftIO) import C2HS.C.Attrs (AttrC, CObj(..), CTag(..), CDef(..), lookupDefObjC, lookupDefTagC, getDefOfIdentC) import C2HS.C.Names (nameAnalysis) import C2HS.C.Trav -- | suffix for files containing C -- csuffix, hsuffix, isuffix :: String csuffix = "c" hsuffix = "h" isuffix = "i" -- | parse a header file, raise an error if parsing failed parseHeader :: InputStream -> Position -> CST s CTranslUnit parseHeader is pos = do ns <- getNameSupply case execParser translUnitP is pos builtinTypeNames ns of Left (ParseError (msgs,pos')) -> raiseError pos' msgs >> return (CTranslUnit [] undefNode) Right (ct,ns') -> setNameSupply ns' >> return ct -- | @bsReplace old new haystack@ replaces occurences of @old@ with -- @new@ in the @haystack@. bsReplace :: BS.ByteString -> BS.ByteString -> BS.ByteString -> BS.ByteString bsReplace old new = go id where go acc hay | BS.null hay = BS.concat (acc []) | otherwise = case BS.breakSubstring old hay of (h,t) | BS.null t -> BS.concat (acc [h]) | otherwise -> go (acc . (h:) . (new:)) (BS.drop n t) n = BS.length old -- | given a file name (with suffix), parse that file as a C header and do the -- static analysis (collect defined names) -- -- * currently, lexical and syntactical errors are reported immediately and -- abort the program; others are reported as part of the fatal error message; -- warnings are returned together with the read unit -- loadAttrC :: String -> CST s (AttrC, String) loadAttrC fname = do -- read file -- traceInfoRead fname -- very hacky dodge of Apple's block syntax in -- type definitions. This simply replaces a block -- type with a function pointer type. The issue -- is that language-c does not support this -- syntax, but frameworks such as OpenCL now use -- it in their headers. let fixBlockTypeDef x | BS.isPrefixOf (BS.pack "typedef ") x = bsReplace (BS.pack "(^") (BS.pack "(*") x | otherwise = x fixLines = BS.unlines . map fixBlockTypeDef . BS.lines contents <- fmap fixLines (liftIO $ readInputStream fname) -- parse -- traceInfoParse header <- parseHeader contents (initPos fname) -- name analysis -- traceInfoNA headerWithAttrs <- nameAnalysis header -- check for errors and finalize -- errs <- errorsPresent if errs then do traceInfoErr errmsgs <- showErrors fatal ("C header contains \ \errors:\n\n" ++ errmsgs) -- fatal error else do traceInfoOK warnmsgs <- showErrors return (headerWithAttrs, warnmsgs) where traceInfoRead fname' = putTraceStr tracePhasesSW ("Attempting to read file `" ++ fname' ++ "'...\n") traceInfoParse = putTraceStr tracePhasesSW ("...parsing `" ++ fname ++ "'...\n") traceInfoNA = putTraceStr tracePhasesSW ("...name analysis of `" ++ fname ++ "'...\n") traceInfoErr = putTraceStr tracePhasesSW ("...error(s) detected in `" ++ fname ++ "'.\n") traceInfoOK = putTraceStr tracePhasesSW ("...successfully loaded `" ++ fname ++ "'.\n")