Skip to content

Commit edd28b1

Browse files
Fix Reify and reflect for lists of references. Fixes #102
1 parent 6c5e35a commit edd28b1

File tree

2 files changed

+9
-2
lines changed

2 files changed

+9
-2
lines changed

jvm/CHANGELOG.md

+4
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,10 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/).
2222
* The `Reify`/`Reflect` instances for `()` is now mapped to
2323
a *serializable* small JVM object. This is a more useful instance
2424
for sparkle users.
25+
* `Reify`/`Reflect` were given additional pre and post conditions.
26+
The lifetimes of the marshaled values are disentangled now, which
27+
fixes memory errors in the instances of `[J ty]`.
28+
[#102](https://github.com/tweag/inline-java/pull/102)
2529

2630
## [0.3.0] - 2017-08-31
2731

jvm/src/Language/Java.hs

+5-2
Original file line numberDiff line numberDiff line change
@@ -469,19 +469,22 @@ class (SingI (Interp a), IsReferenceType (Interp a)) => Interpretation (a :: k)
469469
-- say, unmarshall a Java object to a Haskell value. Unlike coercing, in general
470470
-- reifying induces allocations and copies.
471471
class Interpretation a => Reify a where
472+
-- | Precondition: The input reference is valid only until reify returns.
472473
reify :: J (Interp a) -> IO a
473474

474475
default reify :: (Coercible a, Interp a ~ Ty a) => J (Interp a) -> IO a
475-
reify x = return (unsafeUncoerce (JObject x))
476+
reify x = unsafeUncoerce . JObject <$> (newLocalRef x :: IO (J (Ty a)))
476477

477478
-- | Inject a concrete Haskell value into the space of Java objects. That is to
478479
-- say, marshall a Haskell value to a Java object. Unlike coercing, in general
479480
-- reflection induces allocations and copies.
480481
class Interpretation a => Reflect a where
482+
-- | Postcondition: The input value remains valid if the output reference is
483+
-- deleted.
481484
reflect :: a -> IO (J (Interp a))
482485

483486
default reflect :: (Coercible a, Interp a ~ Ty a) => a -> IO (J (Interp a))
484-
reflect x = return (jobject x)
487+
reflect x = newLocalRef (jobject x)
485488

486489
#if ! (__GLASGOW_HASKELL__ == 800 && __GLASGOW_HASKELL_PATCHLEVEL1__ == 1)
487490
reifyMVector

0 commit comments

Comments
 (0)