{-# LANGUAGE DeriveDataTypeable #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DeriveAnyClass #-}
module Language.Rust.Data.Ident (Ident(..), mkIdent, Name, IdentName(..) ) where
import GHC.Generics ( Generic )
import Control.DeepSeq ( NFData )
import Data.Data ( Data )
import Data.Typeable ( Typeable )
import Data.List ( foldl' )
import Data.Char ( ord )
import Data.String ( IsString(..) )
data Ident
= Ident { name :: IdentName
, raw :: Bool
, hash :: {-# UNPACK #-} !Int
} deriving (Data, Typeable, Generic, NFData)
instance Show Ident where
show = show . name
instance IsString Ident where
fromString = mkIdent
instance Eq Ident where
i1 == i2 = hash i1 == hash i2 && name i1 == name i2 && raw i1 == raw i2
i1 /= i2 = hash i1 /= hash i2 || name i1 /= name i2 || raw i1 /= raw i2
instance Ord Ident where
compare i1 i2 = case compare i1 i2 of
EQ -> compare (raw i1, name i1) (raw i2, name i2)
rt -> rt
instance Monoid Ident where
mappend = (<>)
mempty = mkIdent ""
instance Semigroup Ident where
Ident (Name n1) _ _ <> Ident (Name n2) _ _ = mkIdent (n1 <> n2)
Ident (HaskellName n1) _ _ <> Ident (HaskellName n2) _ _ = mkIdent (n1 <> n2)
Ident (Name n1) _ _ <> Ident (HaskellName n2) _ _ = mkIdent (n1 <> n2)
Ident (HaskellName n1) _ _ <> Ident (Name n2) _ _ = mkIdent (n1 <> n2)
mkIdent :: String -> Ident
mkIdent ('$':'{':'i':'|':ss) = Ident (HaskellName $ init $ init ss) False (hashString $ init $ init ss)
mkIdent s = Ident (Name s) False (hashString s)
hashString :: String -> Int
hashString = foldl' f golden
where f m c = fromIntegral (ord c) * magic + m
magic = 0xdeadbeef
golden = 1013904242
type Name = String
data IdentName =
Name Name
| HaskellName String
deriving (Eq, Ord, Data, Typeable, Generic, NFData)
instance Show IdentName where
show (Name n) = show n
show (HaskellName n) = "${i|" ++ n ++ "|}"