From 3dfd51d2832c465ba0a782aa08c3c6bc29cf4607 Mon Sep 17 00:00:00 2001 From: Oleg Grenrus Date: Fri, 30 Aug 2024 17:25:10 +0300 Subject: [PATCH 1/5] Add enum example --- hs-bindgen/examples/enums.h | 9 +++++++++ hs-bindgen/fixtures/enums.dump.txt | 3 +++ hs-bindgen/tests/golden.hs | 17 +++++++++++------ 3 files changed, 23 insertions(+), 6 deletions(-) create mode 100644 hs-bindgen/examples/enums.h create mode 100644 hs-bindgen/fixtures/enums.dump.txt diff --git a/hs-bindgen/examples/enums.h b/hs-bindgen/examples/enums.h new file mode 100644 index 00000000..8a080476 --- /dev/null +++ b/hs-bindgen/examples/enums.h @@ -0,0 +1,9 @@ +#ifndef ENUMS_H +#define ENUMS_H + +enum first { + FIRST1, + FIRST2 +}; + +#endif /* ENUMS_H */ diff --git a/hs-bindgen/fixtures/enums.dump.txt b/hs-bindgen/fixtures/enums.dump.txt new file mode 100644 index 00000000..afcd1598 --- /dev/null +++ b/hs-bindgen/fixtures/enums.dump.txt @@ -0,0 +1,3 @@ +"first" :: "Enum" + "FIRST1" :: "Int" + "FIRST2" :: "Int" diff --git a/hs-bindgen/tests/golden.hs b/hs-bindgen/tests/golden.hs index f98e923d..2d9f2909 100644 --- a/hs-bindgen/tests/golden.hs +++ b/hs-bindgen/tests/golden.hs @@ -14,16 +14,21 @@ main :: IO () main = do findPackageDirectory "hs-bindgen" defaultMain $ testGroup "golden" - [ goldenVsStringDiff "simple_structs" diff "fixtures/simple_structs.dump.txt" $ do - let fp = "examples/simple_structs.h" - args = [] - res <- getClangAST args fp - - return $ LBS8.pack $ unlines $ concatMap treeToLines res + [ golden "simple_structs" + , golden "enums" ] where diff ref new = ["diff", "-u", ref, new] + golden name = goldenVsStringDiff name diff ("fixtures" (name ++ ".dump.txt")) $ do + -- -<.> does weird stuff for filenames with multiple dots; + -- I usually simply avoid using it. + let fp = "examples" (name ++ ".h") + args = [] + res <- getClangAST args fp + + return $ LBS8.pack $ unlines $ concatMap treeToLines res + treeToLines :: Tree Element -> [String] treeToLines tree = go 0 tree [] where go :: Int -> Tree Element -> [String] -> [String] From be7f0a7cf022da8fec9fb9f346928eef84a63635 Mon Sep 17 00:00:00 2001 From: Oleg Grenrus Date: Fri, 30 Aug 2024 17:40:45 +0300 Subject: [PATCH 2/5] golden tests: generate TH declarations (doesnt do anything atm) --- hs-bindgen/fixtures/enums.th.txt | 0 hs-bindgen/fixtures/simple_structs.th.txt | 0 hs-bindgen/hs-bindgen.cabal | 1 + hs-bindgen/src/HsBindgen/Lib.hs | 1 + hs-bindgen/src/HsBindgen/Util/Tracer.hs | 6 +++++- hs-bindgen/tests/golden.hs | 19 ++++++++++++++++++- 6 files changed, 25 insertions(+), 2 deletions(-) create mode 100644 hs-bindgen/fixtures/enums.th.txt create mode 100644 hs-bindgen/fixtures/simple_structs.th.txt diff --git a/hs-bindgen/fixtures/enums.th.txt b/hs-bindgen/fixtures/enums.th.txt new file mode 100644 index 00000000..e69de29b diff --git a/hs-bindgen/fixtures/simple_structs.th.txt b/hs-bindgen/fixtures/simple_structs.th.txt new file mode 100644 index 00000000..e69de29b diff --git a/hs-bindgen/hs-bindgen.cabal b/hs-bindgen/hs-bindgen.cabal index 16f01c64..115d616e 100644 --- a/hs-bindgen/hs-bindgen.cabal +++ b/hs-bindgen/hs-bindgen.cabal @@ -97,6 +97,7 @@ test-suite golden -- Inherited dependencies , containers , bytestring + , template-haskell build-depends: -- External dependencies , directory ^>=1.3.6.2 diff --git a/hs-bindgen/src/HsBindgen/Lib.hs b/hs-bindgen/src/HsBindgen/Lib.hs index 40cf6fe1..4184d86e 100644 --- a/hs-bindgen/src/HsBindgen/Lib.hs +++ b/hs-bindgen/src/HsBindgen/Lib.hs @@ -39,6 +39,7 @@ module HsBindgen.Lib ( -- * Logging , Tracer + , nullTracer , contramap , PrettyLogMsg(..) , mkTracerIO diff --git a/hs-bindgen/src/HsBindgen/Util/Tracer.hs b/hs-bindgen/src/HsBindgen/Util/Tracer.hs index 8a45efe9..629555d2 100644 --- a/hs-bindgen/src/HsBindgen/Util/Tracer.hs +++ b/hs-bindgen/src/HsBindgen/Util/Tracer.hs @@ -5,6 +5,7 @@ -- Indended for unqualified import. module HsBindgen.Util.Tracer ( Tracer -- opaque + , nullTracer , PrettyLogMsg(..) -- * Using the tracer , Level(..) @@ -42,6 +43,9 @@ traceWith t l a = Contra.traceWith (unwrap t) (l, a) Internal: general tracer construction -------------------------------------------------------------------------------} +nullTracer :: Monad m => Tracer m a +nullTracer = Wrap Contra.nullTracer + mkTracer :: forall m a. Monad m => (a -> m ()) -- ^ Output error @@ -96,4 +100,4 @@ mkTracerQ = -------------------------------------------------------------------------------} class PrettyLogMsg a where - prettyLogMsg :: a -> String \ No newline at end of file + prettyLogMsg :: a -> String diff --git a/hs-bindgen/tests/golden.hs b/hs-bindgen/tests/golden.hs index 2d9f2909..c71aa91f 100644 --- a/hs-bindgen/tests/golden.hs +++ b/hs-bindgen/tests/golden.hs @@ -6,6 +6,7 @@ import System.Directory (doesFileExist, setCurrentDirectory) import System.FilePath ((), (-<.>)) import Test.Tasty (defaultMain, testGroup) import Test.Tasty.Golden (goldenVsStringDiff) +import Language.Haskell.TH.Ppr (ppr) import HsBindgen.Clang.Util.Classification import HsBindgen.Lib @@ -20,7 +21,12 @@ main = do where diff ref new = ["diff", "-u", ref, new] - golden name = goldenVsStringDiff name diff ("fixtures" (name ++ ".dump.txt")) $ do + golden name = testGroup name + [ goldenDump name + , goldenTH name + ] + + goldenDump name = goldenVsStringDiff "ast" diff ("fixtures" (name ++ ".dump.txt")) $ do -- -<.> does weird stuff for filenames with multiple dots; -- I usually simply avoid using it. let fp = "examples" (name ++ ".h") @@ -29,6 +35,17 @@ main = do return $ LBS8.pack $ unlines $ concatMap treeToLines res + goldenTH name = goldenVsStringDiff "th" diff ("fixtures" (name ++ ".th.txt")) $ do + -- -<.> does weird stuff for filenames with multiple dots; + -- I usually simply avoid using it. + let fp = "examples" (name ++ ".h") + args = [] + + header <- parseCHeader nullTracer args fp + let decls = genDecls header + + return $ LBS8.pack $ unlines $ map (show . ppr) decls + treeToLines :: Tree Element -> [String] treeToLines tree = go 0 tree [] where go :: Int -> Tree Element -> [String] -> [String] From 1a130f117a74b7d211b476cc73e6ad3e87f24985 Mon Sep 17 00:00:00 2001 From: Oleg Grenrus Date: Sun, 1 Sep 2024 15:13:05 +0300 Subject: [PATCH 3/5] Add tree-diff test for CHeader output --- hs-bindgen/fixtures/enums.tree-diff.txt | 1 + .../fixtures/simple_structs.tree-diff.txt | 69 +++++++++++++++++++ hs-bindgen/hs-bindgen.cabal | 4 +- hs-bindgen/src/HsBindgen/Lib.hs | 2 + hs-bindgen/tests/Orphans.hs | 16 +++++ hs-bindgen/tests/golden.hs | 12 ++++ 6 files changed, 103 insertions(+), 1 deletion(-) create mode 100644 hs-bindgen/fixtures/enums.tree-diff.txt create mode 100644 hs-bindgen/fixtures/simple_structs.tree-diff.txt create mode 100644 hs-bindgen/tests/Orphans.hs diff --git a/hs-bindgen/fixtures/enums.tree-diff.txt b/hs-bindgen/fixtures/enums.tree-diff.txt new file mode 100644 index 00000000..c4784bca --- /dev/null +++ b/hs-bindgen/fixtures/enums.tree-diff.txt @@ -0,0 +1 @@ +WrapCHeader (Header []) diff --git a/hs-bindgen/fixtures/simple_structs.tree-diff.txt b/hs-bindgen/fixtures/simple_structs.tree-diff.txt new file mode 100644 index 00000000..9e99cfd2 --- /dev/null +++ b/hs-bindgen/fixtures/simple_structs.tree-diff.txt @@ -0,0 +1,69 @@ +WrapCHeader + (Header + [ + DeclStruct + Struct { + structTag = Just "S1", + structSizeof = 8, + structAlignment = 4, + structFields = [ + StructField { + fieldName = "a", + fieldType = PrimInt}, + StructField { + fieldName = "b", + fieldType = PrimChar}]}, + DeclStruct + Struct { + structTag = Just "S2", + structSizeof = 12, + structAlignment = 4, + structFields = [ + StructField { + fieldName = "a", + fieldType = PrimChar}, + StructField { + fieldName = "b", + fieldType = PrimInt}, + StructField { + fieldName = "c", + fieldType = PrimFloat}]}, + DeclTypedef + Typedef { + typedefName = "S2_t", + typedefType = TypStruct + Struct { + structTag = Just "S2", + structSizeof = 12, + structAlignment = 4, + structFields = [ + StructField { + fieldName = "a", + fieldType = PrimChar}, + StructField { + fieldName = "b", + fieldType = PrimInt}, + StructField { + fieldName = "c", + fieldType = PrimFloat}]}}, + DeclStruct + Struct { + structTag = Nothing, + structSizeof = 1, + structAlignment = 1, + structFields = [ + StructField { + fieldName = "a", + fieldType = PrimChar}]}, + DeclTypedef + Typedef { + typedefName = "S3_t", + typedefType = TypStruct + Struct { + structTag = Nothing, + structSizeof = 1, + structAlignment = 1, + structFields = [ + StructField { + fieldName = "a", + fieldType = PrimChar}]}}]) diff --git a/hs-bindgen/hs-bindgen.cabal b/hs-bindgen/hs-bindgen.cabal index 115d616e..2d4ce34d 100644 --- a/hs-bindgen/hs-bindgen.cabal +++ b/hs-bindgen/hs-bindgen.cabal @@ -40,7 +40,7 @@ library exposed-modules: HsBindgen.Lib HsBindgen.TH - other-modules: + exposed-modules: HsBindgen.C.AST HsBindgen.C.Parser HsBindgen.Hs.Annotation @@ -89,6 +89,7 @@ test-suite golden type: exitcode-stdio-1.0 main-is: golden.hs hs-source-dirs: tests + other-modules: Orphans build-depends: -- Internal dependencies , hs-bindgen @@ -104,3 +105,4 @@ test-suite golden , filepath ^>=1.4.2.2 || ^>=1.5.2.0 , tasty ^>=1.5 , tasty-golden ^>=2.3.5 + , tree-diff ^>=0.3.1 diff --git a/hs-bindgen/src/HsBindgen/Lib.hs b/hs-bindgen/src/HsBindgen/Lib.hs index 4184d86e..00d4a29c 100644 --- a/hs-bindgen/src/HsBindgen/Lib.hs +++ b/hs-bindgen/src/HsBindgen/Lib.hs @@ -46,6 +46,7 @@ module HsBindgen.Lib ( ) where import Data.Tree (Forest) +import GHC.Generics (Generic) import Language.Haskell.Exts qualified as Hs import Language.Haskell.Meta qualified as Meta import Language.Haskell.TH qualified as TH @@ -71,6 +72,7 @@ import HsBindgen.Util.Tracer newtype CHeader = WrapCHeader { unwrapCHeader :: C.Header } + deriving (Eq, Generic) newtype HsModule = WrapHsModule { unwrapHsModule :: Hs.Module Ann diff --git a/hs-bindgen/tests/Orphans.hs b/hs-bindgen/tests/Orphans.hs new file mode 100644 index 00000000..a0a54c92 --- /dev/null +++ b/hs-bindgen/tests/Orphans.hs @@ -0,0 +1,16 @@ +{-# OPTIONS_GHC -Wno-orphans #-} +module Orphans where + +import Data.TreeDiff.Class (ToExpr) + +import HsBindgen.Lib +import HsBindgen.C.AST qualified as C + +instance ToExpr CHeader +instance ToExpr C.Header +instance ToExpr C.Decl +instance ToExpr C.Struct +instance ToExpr C.Typedef +instance ToExpr C.Typ +instance ToExpr C.PrimType +instance ToExpr C.StructField diff --git a/hs-bindgen/tests/golden.hs b/hs-bindgen/tests/golden.hs index c71aa91f..74f62611 100644 --- a/hs-bindgen/tests/golden.hs +++ b/hs-bindgen/tests/golden.hs @@ -6,7 +6,11 @@ import System.Directory (doesFileExist, setCurrentDirectory) import System.FilePath ((), (-<.>)) import Test.Tasty (defaultMain, testGroup) import Test.Tasty.Golden (goldenVsStringDiff) +import Test.Tasty.Golden.Advanced (goldenTest) import Language.Haskell.TH.Ppr (ppr) +import Data.TreeDiff.Golden (ediffGolden) + +import Orphans () import HsBindgen.Clang.Util.Classification import HsBindgen.Lib @@ -23,6 +27,7 @@ main = do golden name = testGroup name [ goldenDump name + , goldenTreeDiff name , goldenTH name ] @@ -35,6 +40,13 @@ main = do return $ LBS8.pack $ unlines $ concatMap treeToLines res + goldenTreeDiff name = ediffGolden goldenTest "treediff" ("fixtures" (name ++ ".tree-diff.txt")) $ do + let fp = "examples" (name ++ ".h") + args = [] + + header <- parseCHeader nullTracer args fp + return header + goldenTH name = goldenVsStringDiff "th" diff ("fixtures" (name ++ ".th.txt")) $ do -- -<.> does weird stuff for filenames with multiple dots; -- I usually simply avoid using it. From f342f390f0181b05ec873f3db8506b09a498cedc Mon Sep 17 00:00:00 2001 From: Oleg Grenrus Date: Sun, 1 Sep 2024 15:38:06 +0300 Subject: [PATCH 4/5] Add test for primitive types (not all are yet supported) --- hs-bindgen/examples/primitive_types.h | 40 +++++++++++++++++++ hs-bindgen/fixtures/primitive_types.dump.txt | 30 ++++++++++++++ hs-bindgen/fixtures/primitive_types.th.txt | 0 .../fixtures/primitive_types.tree-diff.txt | 24 +++++++++++ hs-bindgen/tests/golden.hs | 1 + 5 files changed, 95 insertions(+) create mode 100644 hs-bindgen/examples/primitive_types.h create mode 100644 hs-bindgen/fixtures/primitive_types.dump.txt create mode 100644 hs-bindgen/fixtures/primitive_types.th.txt create mode 100644 hs-bindgen/fixtures/primitive_types.tree-diff.txt diff --git a/hs-bindgen/examples/primitive_types.h b/hs-bindgen/examples/primitive_types.h new file mode 100644 index 00000000..dbfbabaa --- /dev/null +++ b/hs-bindgen/examples/primitive_types.h @@ -0,0 +1,40 @@ +struct primitive { + char c; + signed char sc; + unsigned char uc; + + short s; + short int si; + signed short ss; + signed short int ssi; + + unsigned short us; + unsigned short int usi; + + int i; + signed s2; + signed int si2; + + unsigned u; + unsigned int ui; + + long l; + long int li; + signed long sl; + signed long int sli; + + unsigned long ul; + unsigned long int uli; + + long long ll; + long long int lli; + signed long long sll; + signed long long int slli; + + unsigned long long ull; + unsigned long long int ulli; + + float f; + double d; + long double ld; +}; \ No newline at end of file diff --git a/hs-bindgen/fixtures/primitive_types.dump.txt b/hs-bindgen/fixtures/primitive_types.dump.txt new file mode 100644 index 00000000..2c481fff --- /dev/null +++ b/hs-bindgen/fixtures/primitive_types.dump.txt @@ -0,0 +1,30 @@ +"primitive" :: "Record" + "c" :: "Char_S" + "sc" :: "SChar" + "uc" :: "UChar" + "s" :: "Short" + "si" :: "Short" + "ss" :: "Short" + "ssi" :: "Short" + "us" :: "UShort" + "usi" :: "UShort" + "i" :: "Int" + "s2" :: "Int" + "si2" :: "Int" + "u" :: "UInt" + "ui" :: "UInt" + "l" :: "Long" + "li" :: "Long" + "sl" :: "Long" + "sli" :: "Long" + "ul" :: "ULong" + "uli" :: "ULong" + "ll" :: "LongLong" + "lli" :: "LongLong" + "sll" :: "LongLong" + "slli" :: "LongLong" + "ull" :: "ULongLong" + "ulli" :: "ULongLong" + "f" :: "Float" + "d" :: "Double" + "ld" :: "LongDouble" diff --git a/hs-bindgen/fixtures/primitive_types.th.txt b/hs-bindgen/fixtures/primitive_types.th.txt new file mode 100644 index 00000000..e69de29b diff --git a/hs-bindgen/fixtures/primitive_types.tree-diff.txt b/hs-bindgen/fixtures/primitive_types.tree-diff.txt new file mode 100644 index 00000000..c7d275c2 --- /dev/null +++ b/hs-bindgen/fixtures/primitive_types.tree-diff.txt @@ -0,0 +1,24 @@ +WrapCHeader + (Header + [ + DeclStruct + Struct { + structTag = Just "primitive", + structSizeof = 176, + structAlignment = 16, + structFields = [ + StructField { + fieldName = "c", + fieldType = PrimChar}, + StructField { + fieldName = "i", + fieldType = PrimInt}, + StructField { + fieldName = "s2", + fieldType = PrimInt}, + StructField { + fieldName = "si2", + fieldType = PrimInt}, + StructField { + fieldName = "f", + fieldType = PrimFloat}]}]) diff --git a/hs-bindgen/tests/golden.hs b/hs-bindgen/tests/golden.hs index 74f62611..3b017428 100644 --- a/hs-bindgen/tests/golden.hs +++ b/hs-bindgen/tests/golden.hs @@ -21,6 +21,7 @@ main = do defaultMain $ testGroup "golden" [ golden "simple_structs" , golden "enums" + , golden "primitive_types" ] where diff ref new = ["diff", "-u", ref, new] From 1bbaf6a2bc73faf857d80f43525ae3ffcac13c2c Mon Sep 17 00:00:00 2001 From: Oleg Grenrus Date: Sun, 1 Sep 2024 17:38:49 +0300 Subject: [PATCH 5/5] Hardcode target in tests --- hs-bindgen/tests/golden.hs | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/hs-bindgen/tests/golden.hs b/hs-bindgen/tests/golden.hs index 3b017428..2b48eeb2 100644 --- a/hs-bindgen/tests/golden.hs +++ b/hs-bindgen/tests/golden.hs @@ -36,14 +36,15 @@ main = do -- -<.> does weird stuff for filenames with multiple dots; -- I usually simply avoid using it. let fp = "examples" (name ++ ".h") - args = [] + args = ["-target", "x86_64-pc-linux-gnu"] + res <- getClangAST args fp return $ LBS8.pack $ unlines $ concatMap treeToLines res goldenTreeDiff name = ediffGolden goldenTest "treediff" ("fixtures" (name ++ ".tree-diff.txt")) $ do let fp = "examples" (name ++ ".h") - args = [] + args = ["-target", "x86_64-pc-linux-gnu"] header <- parseCHeader nullTracer args fp return header @@ -52,7 +53,7 @@ main = do -- -<.> does weird stuff for filenames with multiple dots; -- I usually simply avoid using it. let fp = "examples" (name ++ ".h") - args = [] + args = ["-target", "x86_64-pc-linux-gnu"] header <- parseCHeader nullTracer args fp let decls = genDecls header