Skip to content

Commit

Permalink
Handle interface methods in LF conversion (#11054)
Browse files Browse the repository at this point in the history
changelog_begin
changelog_end
  • Loading branch information
cocreature authored Sep 29, 2021
1 parent 87ecf1f commit 6d8cf70
Show file tree
Hide file tree
Showing 3 changed files with 110 additions and 14 deletions.
73 changes: 60 additions & 13 deletions compiler/damlc/daml-lf-conversion/src/DA/Daml/LFConversion.hs
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
-- Copyright (c) 2021 Digital Asset (Switzerland) GmbH and/or its affiliates. All rights reserved.
-- SPDX-License-Identifier: Apache-2.0

{-# LANGUAGE MultiWayIf #-}
{-# OPTIONS_GHC -Wno-unused-matches #-}
{-# OPTIONS_GHC -Wno-incomplete-patterns #-}
Expand Down Expand Up @@ -91,7 +90,7 @@ import Development.IDE.Types.Diagnostics
import Development.IDE.Types.Location
import Development.IDE.GHC.Util

import Control.Lens
import Control.Lens hiding (MethodName)
import Control.Monad.Except
import Control.Monad.Extra
import Control.Monad.Reader
Expand Down Expand Up @@ -175,7 +174,8 @@ data Env = Env
,envTemplateBinds :: MS.Map TypeConName TemplateBinds
,envExceptionBinds :: MS.Map TypeConName ExceptionBinds
,envChoiceData :: MS.Map TypeConName [ChoiceData]
,envImplements :: MS.Map TypeConName [GHC.Type]
,envImplements :: MS.Map TypeConName [GHC.TyCon]
,envInterfaceInstances :: MS.Map (GHC.Module, TypeConName, TypeConName) (GHC.Expr GHC.CoreBndr)
,envInterfaces :: S.Set TypeConName
,envIsGenerated :: Bool
,envTypeVars :: !(MS.Map Var TypeVarName)
Expand Down Expand Up @@ -432,11 +432,13 @@ convertInterfaces env tyThings = interfaceClasses
[ convertChoice arg res
|
TypeCon (NameIn DA_Internal_Template_Functions "HasExercise") [_, arg, res] <- classSCTheta cls]
-- Drop toIface/fromIface/toIfaceContractId/fromIfaceContractId to get only user-defined methods.
methods <- mapM convertMethod (drop 4 $ classMethods cls)
pure DefInterface
{ intLocation = Nothing
, intName = mkTypeCon [name]
, intChoices = NM.fromList choices
, intMethods = NM.empty -- TODO https://github.com/digital-asset/daml/issues/11006
, intMethods = NM.fromList methods
}
convertChoice :: TyCoRep.Type -> TyCoRep.Type -> ConvertM InterfaceChoice
convertChoice arg res = do
Expand All @@ -449,6 +451,17 @@ convertInterfaces env tyThings = interfaceClasses
, ifcArgType = arg
, ifcRetType = res
}
convertMethod :: Var -> ConvertM InterfaceMethod
convertMethod method = do
retTy <- convertType env (varType method) >>= \case
TForall _ (_dict :-> _iface :-> retTy) -> pure retTy
ty -> unsupported "Interface method must be a function" (varType method)
let methodName = occNameString (getOccName (varName method))
pure InterfaceMethod
{ ifmLocation = Nothing
, ifmName = MethodName (T.pack methodName)
, ifmType = retTy
}

convertModule
:: LF.Version
Expand Down Expand Up @@ -486,7 +499,16 @@ convertModule lfVersion pkgMap stablePackages isGenerated file x details = runCo
[ (mkTypeCon [getOccText tplTy], [ifaceTy])
| (name, _) <- binds
, "_implements_" `T.isPrefixOf` getOccText name
, TypeCon _ [TypeCon tplTy [], ifaceTy] <- [varType name]
, TypeCon _ [TypeCon tplTy [], TypeCon ifaceTy []] <- [varType name]
]
tplInterfaceInstances :: MS.Map (GHC.Module, TypeConName, TypeConName) (GHC.Expr GHC.CoreBndr)
tplInterfaceInstances = MS.fromList
[ ((mod, mkTypeCon [iface], mkTypeCon [getOccText tpl]), val)
| (name, val) <- binds
, DFunId _ <- [idDetails name]
, TypeCon ifaceCls [TypeCon tpl []] <- [varType name]
, Just iface <- [T.stripPrefix "Is" $ getOccText ifaceCls]
, Just mod <- [nameModule_maybe (getName ifaceCls)]
]
choiceData = MS.fromListWith (++)
[ (mkTypeCon [getOccText tplTy], [ChoiceData ty v])
Expand All @@ -513,6 +535,7 @@ convertModule lfVersion pkgMap stablePackages isGenerated file x details = runCo
, envTemplateBinds = templateBinds
, envExceptionBinds = exceptionBinds
, envImplements = tplImplements
, envInterfaceInstances = tplInterfaceInstances
, envChoiceData = choiceData
, envIsGenerated = isGenerated
, envTypeVars = MS.empty
Expand Down Expand Up @@ -829,17 +852,30 @@ useSingleMethodDict env x _ =
unhandled "useSingleMethodDict: not a single method type class dictionary" x

convertImplements :: Env -> LF.TypeConName -> ConvertM (NM.NameMap TemplateImplements)
convertImplements env tplTypeCon = NM.fromList . map stub <$>
mapM convertInterfaceCon (MS.findWithDefault [] tplTypeCon (envImplements env))
convertImplements env tplTypeCon = NM.fromList <$>
mapM convertInterface (MS.findWithDefault [] tplTypeCon (envImplements env))
where
stub tcon = TemplateImplements tcon NM.empty
-- TODO https://github.com/digital-asset/daml/issues/11006
-- convert methods
convertInterfaceCon ty = do
ty' <- convertType env ty
case ty' of
convertInterface ty = do
ty' <- convertTyCon env ty
con <- case ty' of
TCon con -> pure con
_ -> unhandled "interface type" ty
let mod = nameModule (getName ty)
dictExpr <- case MS.lookup (mod, qualObject con, tplTypeCon) (envInterfaceInstances env) of
Just e -> pure e
Nothing -> unhandled ("missing interface instance for " <> show con) ()
fields <- convertExpr env dictExpr >>= \case
EStructCon fields -> pure fields
e -> unhandled ("Expected struct for interface dict but got " <> show e) ()
-- Drop superclass constraints and to/fromIface & to/fromIfaceContractId
-- which are always at the beginning.
let methodFields = drop 4 (filter (\(FieldName f, _) -> "m_" `T.isPrefixOf` f) fields)
let methods = NM.fromList
[ TemplateImplementsMethod (MethodName methodName) e
| (FieldName fieldName, e) <- methodFields
, Just methodName <- [T.stripPrefix "m_" fieldName]
]
pure (TemplateImplements con methods)

convertChoices :: Env -> LF.TypeConName -> TemplateBinds -> ConvertM (NM.NameMap TemplateChoice)
convertChoices env tplTypeCon tbinds =
Expand Down Expand Up @@ -1050,6 +1086,17 @@ convertExpr env0 e = do
pure $ ETmLam (v, TStruct fields) $ ERecCon tupleType $ zipWithFrom mkFieldProj (1 :: Int) fields
go env (VarIn GHC_Types "primitive") (LType (isStrLitTy -> Just y) : LType t : args)
= fmap (, args) $ convertPrim (envLfVersion env) (unpackFS y) <$> convertType env t
go env (VarIn GHC_Types "primitiveInterface") (LType (isStrLitTy -> Just y) : LType t : args)
= do
ty <- convertType env t
case ty of
TCon iface :-> _ ->
pure
( ETmLam (mkVar "i", TCon iface) $
ECallInterface iface (MethodName $ T.pack $ unpackFS y) (EVar $ mkVar "i")
, args
)
_ -> unsupported "primitiveInterface was not applied to function from interface" t
-- NOTE(MH): `getFieldPrim` and `setFieldPrim` are used by the record
-- preprocessor to magically implement the `HasField` instances for records.
go env (VarIn DA_Internal_Record "getFieldPrim") (LType (isStrLitTy -> Just name) : LType record : LType _field : args) = do
Expand Down
5 changes: 4 additions & 1 deletion compiler/damlc/daml-prim-src/GHC/Types.daml
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ module GHC.Types (
Text, Decimal,
Opaque,
ifThenElse,
primitive, magic, external,
primitive, primitiveInterface, magic, external,
DamlEnum,
DamlInterface,

Expand Down Expand Up @@ -149,6 +149,9 @@ external = external --deleted by the compiler
primitive : forall (f : Symbol) b. b
primitive = primitive -- deleted by the compiler

primitiveInterface : forall (f : Symbol) b. b
primitiveInterface = primitiveInterface -- deleted by the compiler

-- | HIDE Handled actually in the guts of the compiler
magic : forall (f : Symbol) b. b
magic = magic -- deleted by the compiler
Expand Down
46 changes: 46 additions & 0 deletions compiler/damlc/tests/daml-test-files/InterfaceMethods.daml
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
-- Copyright (c) 2021 Digital Asset (Switzerland) GmbH and/or its affiliates. All rights reserved.
-- SPDX-License-Identifier: Apache-2.0

-- @SINCE-LF-FEATURE DAML_INTERFACE

-- TODO https://github.com/digital-asset/daml/issues/11006
-- Merge into Interface once methods are supported on the scala side.
-- @ERROR Expr.call_interface not yet implemented
module InterfaceMethods where

interface Token where
getAmount : Decimal
choice Split : (ContractId Token, ContractId Token)
with
splitAmount : Decimal

choice Transfer : ContractId Token
with
newOwner : Party

template Asset
with
issuer : Party
owner : Party
amount : Decimal
where
signatory issuer, owner
implements Token where
let getAmount = amount
choice Split : (ContractId Token, ContractId Token)
with
splitAmount : Decimal
controller owner
do
assert (splitAmount < amount)
cid1 <- create this with amount = splitAmount
cid2 <- create this with amount = amount - splitAmount
pure (toTokenContractId cid1, toTokenContractId cid2)

choice Transfer : ContractId Token
with
newOwner : Party
controller owner, newOwner
do
cid <- create this with owner = newOwner
pure (toTokenContractId cid)

0 comments on commit 6d8cf70

Please sign in to comment.