diff options
author | Zahary Karadjov <zahary@gmail.com> | 2017-04-07 19:35:05 +0300 |
---|---|---|
committer | Zahary Karadjov <zahary@gmail.com> | 2017-04-07 19:35:05 +0300 |
commit | e11b3520ff54387df5016afc075d2a7cc178031b (patch) | |
tree | 31f114e3badcbbabf488175f561dbe178fa6cc22 | |
parent | fb3ff64450526f532492c04c6cc02469cfb3d633 (diff) | |
download | Nim-e11b3520ff54387df5016afc075d2a7cc178031b.tar.gz |
fix #5654
-rw-r--r-- | compiler/ccgexprs.nim | 2 | ||||
-rw-r--r-- | compiler/ccgtypes.nim | 5 | ||||
-rw-r--r-- | tests/concepts/trandom_vars.nim | 42 |
3 files changed, 47 insertions, 2 deletions
diff --git a/compiler/ccgexprs.nim b/compiler/ccgexprs.nim index 15bd7f2e0..f981f9595 100644 --- a/compiler/ccgexprs.nim +++ b/compiler/ccgexprs.nim @@ -267,7 +267,7 @@ proc genAssignment(p: BProc, dest, src: TLoc, flags: TAssignmentFlags) = # little HACK to support the new 'var T' as return type: linefmt(p, cpsStmts, "$1 = $2;$n", rdLoc(dest), rdLoc(src)) return - let ty = skipTypes(dest.t, abstractRange) + let ty = skipTypes(dest.t, abstractRange + tyUserTypeClasses) case ty.kind of tyRef: genRefAssign(p, dest, src, flags) diff --git a/compiler/ccgtypes.nim b/compiler/ccgtypes.nim index 9915ad355..f1e32fd1e 100644 --- a/compiler/ccgtypes.nim +++ b/compiler/ccgtypes.nim @@ -164,7 +164,7 @@ proc mapType(typ: PType): TCTypeKind = of tySet: result = mapSetType(typ) of tyOpenArray, tyArray, tyVarargs: result = ctArray of tyObject, tyTuple: result = ctStruct - of tyUserTypeClass, tyUserTypeClassInst: + of tyUserTypeClasses: internalAssert typ.isResolvedUserTypeClass return mapType(typ.lastSon) of tyGenericBody, tyGenericInst, tyGenericParam, tyDistinct, tyOrdinal, @@ -1131,6 +1131,9 @@ proc genTypeInfo(m: BModule, t: PType): Rope = of tyStatic: if t.n != nil: result = genTypeInfo(m, lastSon t) else: internalError("genTypeInfo(" & $t.kind & ')') + of tyUserTypeClasses: + internalAssert t.isResolvedUserTypeClass + return genTypeInfo(m, t.lastSon) of tyProc: if t.callConv != ccClosure: genTypeInfoAuxBase(m, t, t, result, rope"0") diff --git a/tests/concepts/trandom_vars.nim b/tests/concepts/trandom_vars.nim new file mode 100644 index 000000000..a236cebad --- /dev/null +++ b/tests/concepts/trandom_vars.nim @@ -0,0 +1,42 @@ +discard """ +output: "11.0" +""" + +type + # A random number generator + Random = object + random: proc(): float + # A generic typeclass for a random var + RandomVar[A] = concept x + var rng: Random + rng.sample(x) is A + # A few concrete instances + Uniform = object + a, b: float + ClosureVar[A] = object + f: proc(rng: var Random): A + +# How to sample from various concrete instances +proc sample(rng: var Random, u: Uniform): float = u.a + (u.b - u.a) * rng.random() + +proc sample[A](rng: var Random, c: ClosureVar[A]): A = c.f(rng) + +proc uniform(a, b: float): Uniform = Uniform(a: a, b: b) + +# How to lift a function on values to a function on random variables +proc map[A, B](x: RandomVar[A], f: proc(a: A): B): ClosureVar[B] = + proc inner(rng: var Random): B = + f(rng.sample(x)) + + result.f = inner + +import future + +proc fakeRandom(): Random = + result.random = () => 0.5 + +let x = uniform(1, 10).map((x: float) => 2 * x) + +var rng = fakeRandom() + +echo rng.sample(x) |