diff options
author | Vincent Ambo <mail@tazj.in> | 2022-10-22T13·51+0300 |
---|---|---|
committer | tazjin <tazjin@tvl.su> | 2022-10-23T15·50+0000 |
commit | 60f24c3c53eae116bb3051f02f05b74a8ecf37af (patch) | |
tree | 794d987b22303cfbbb1dd8d251e81ab15d67ad68 /tvix/eval/src/value/thunk.rs | |
parent | 4ff06ba67dbe5397a97c2bae78e25d0ab8c026a3 (diff) |
fix(tvix/eval): detect cycles when printing infinite values r/5178
Using the same method as in Thunk::deep_force, detect cycles when printing values by maintaining a set of already seen thunks. With this, display of infinite values matches that of Nix: > nix-instantiate --eval --strict -E 'let as = { x = 123; y = as; }; in as' { x = 123; y = { x = 123; y = <CYCLE>; }; } > tvix-eval -E 'let as = { x = 123; y = as; }; in as' => { x = 123; y = { x = 123; y = <CYCLE>; }; } :: set Change-Id: I007b918d5131d82c28884e46e46ff365ef691aa8 Reviewed-on: https://cl.tvl.fyi/c/depot/+/7056 Tested-by: BuildkiteCI Reviewed-by: grfn <grfn@gws.fyi>
Diffstat (limited to 'tvix/eval/src/value/thunk.rs')
-rw-r--r-- | tvix/eval/src/value/thunk.rs | 13 |
1 files changed, 8 insertions, 5 deletions
diff --git a/tvix/eval/src/value/thunk.rs b/tvix/eval/src/value/thunk.rs index 7bad7e8777ba..bcf8e2fc8756 100644 --- a/tvix/eval/src/value/thunk.rs +++ b/tvix/eval/src/value/thunk.rs @@ -21,7 +21,6 @@ use std::{ cell::{Ref, RefCell, RefMut}, collections::HashSet, - fmt::Display, rc::Rc, }; @@ -35,7 +34,7 @@ use crate::{ Value, }; -use super::Lambda; +use super::{Lambda, TotalDisplay}; /// Internal representation of the different states of a thunk. /// @@ -188,11 +187,15 @@ impl Thunk { } } -impl Display for Thunk { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { +impl TotalDisplay for Thunk { + fn total_fmt(&self, f: &mut std::fmt::Formatter<'_>, set: &mut ThunkSet) -> std::fmt::Result { + if !set.insert(self) { + return f.write_str("<CYCLE>"); + } + match self.0.try_borrow() { Ok(repr) => match &*repr { - ThunkRepr::Evaluated(v) => v.fmt(f), + ThunkRepr::Evaluated(v) => v.total_fmt(f, set), _ => f.write_str("internal[thunk]"), }, |