diff options
-rw-r--r-- | src/yellow/mod.rs | 31 | ||||
-rw-r--r-- | src/yellow/red.rs | 10 | ||||
-rw-r--r-- | src/yellow/syntax.rs | 19 |
3 files changed, 15 insertions, 45 deletions
diff --git a/src/yellow/mod.rs b/src/yellow/mod.rs index 9a6203cc1..4fd0d24d7 100644 --- a/src/yellow/mod.rs +++ b/src/yellow/mod.rs | |||
@@ -3,10 +3,6 @@ mod red; | |||
3 | mod syntax; | 3 | mod syntax; |
4 | mod builder; | 4 | mod builder; |
5 | 5 | ||
6 | use std::{ | ||
7 | sync::{Arc, Weak}, | ||
8 | mem | ||
9 | }; | ||
10 | pub(crate) use self::{ | 6 | pub(crate) use self::{ |
11 | green::{GreenNode, GreenNodeBuilder}, | 7 | green::{GreenNode, GreenNodeBuilder}, |
12 | red::RedNode, | 8 | red::RedNode, |
@@ -14,30 +10,3 @@ pub(crate) use self::{ | |||
14 | builder::GreenBuilder, | 10 | builder::GreenBuilder, |
15 | }; | 11 | }; |
16 | pub use self::syntax::SyntaxNode; | 12 | pub use self::syntax::SyntaxNode; |
17 | |||
18 | // This could be just `*const T`, but we use `Weak` for additional checks | ||
19 | #[derive(Debug)] | ||
20 | pub(crate) struct Ptr<T>(Weak<T>); | ||
21 | |||
22 | impl<T> Clone for Ptr<T> { | ||
23 | fn clone(&self) -> Self { | ||
24 | Ptr(self.0.clone()) | ||
25 | } | ||
26 | } | ||
27 | |||
28 | impl<T> Ptr<T> { | ||
29 | fn clone(self_: &Ptr<T>) -> Ptr<T> { | ||
30 | Ptr(Weak::clone(&self_.0)) | ||
31 | } | ||
32 | |||
33 | fn new(arc: &Arc<T>) -> Ptr<T> { | ||
34 | Ptr(Arc::downgrade(arc)) | ||
35 | } | ||
36 | |||
37 | unsafe fn get(&self) -> &T { | ||
38 | let t = self.0.upgrade() | ||
39 | .expect("caller must guarantee that Ptr is not null"); | ||
40 | let t: &T = &*t; | ||
41 | mem::transmute(t) | ||
42 | } | ||
43 | } | ||
diff --git a/src/yellow/red.rs b/src/yellow/red.rs index e2dbceeae..3fdbfe0c5 100644 --- a/src/yellow/red.rs +++ b/src/yellow/red.rs | |||
@@ -1,7 +1,7 @@ | |||
1 | use std::sync::{Arc, RwLock}; | 1 | use std::sync::{Arc, RwLock}; |
2 | use { | 2 | use { |
3 | TextUnit, | 3 | TextUnit, |
4 | yellow::{Ptr, GreenNode}, | 4 | yellow::GreenNode, |
5 | }; | 5 | }; |
6 | 6 | ||
7 | #[derive(Debug)] | 7 | #[derive(Debug)] |
@@ -13,7 +13,7 @@ pub(crate) struct RedNode { | |||
13 | 13 | ||
14 | #[derive(Debug)] | 14 | #[derive(Debug)] |
15 | struct ParentData { | 15 | struct ParentData { |
16 | parent: Ptr<RedNode>, | 16 | parent: *const RedNode, |
17 | start_offset: TextUnit, | 17 | start_offset: TextUnit, |
18 | index_in_parent: usize, | 18 | index_in_parent: usize, |
19 | } | 19 | } |
@@ -27,7 +27,7 @@ impl RedNode { | |||
27 | 27 | ||
28 | fn new_child( | 28 | fn new_child( |
29 | green: GreenNode, | 29 | green: GreenNode, |
30 | parent: Ptr<RedNode>, | 30 | parent: *const RedNode, |
31 | start_offset: TextUnit, | 31 | start_offset: TextUnit, |
32 | index_in_parent: usize, | 32 | index_in_parent: usize, |
33 | ) -> RedNode { | 33 | ) -> RedNode { |
@@ -62,7 +62,7 @@ impl RedNode { | |||
62 | self.green.children().len() | 62 | self.green.children().len() |
63 | } | 63 | } |
64 | 64 | ||
65 | pub(crate) fn nth_child(&self, me: Ptr<RedNode>, idx: usize) -> Arc<RedNode> { | 65 | pub(crate) fn nth_child(&self, idx: usize) -> Arc<RedNode> { |
66 | match &self.children.read().unwrap()[idx] { | 66 | match &self.children.read().unwrap()[idx] { |
67 | Some(child) => return child.clone(), | 67 | Some(child) => return child.clone(), |
68 | None => (), | 68 | None => (), |
@@ -72,7 +72,7 @@ impl RedNode { | |||
72 | let green_children = self.green.children(); | 72 | let green_children = self.green.children(); |
73 | let start_offset = self.start_offset() | 73 | let start_offset = self.start_offset() |
74 | + green_children[..idx].iter().map(|x| x.text_len()).sum::<TextUnit>(); | 74 | + green_children[..idx].iter().map(|x| x.text_len()).sum::<TextUnit>(); |
75 | let child = RedNode::new_child(green_children[idx].clone(), me, start_offset, idx); | 75 | let child = RedNode::new_child(green_children[idx].clone(), self, start_offset, idx); |
76 | children[idx] = Some(Arc::new(child)) | 76 | children[idx] = Some(Arc::new(child)) |
77 | } | 77 | } |
78 | children[idx].as_ref().unwrap().clone() | 78 | children[idx].as_ref().unwrap().clone() |
diff --git a/src/yellow/syntax.rs b/src/yellow/syntax.rs index 53d8c82b9..fa51e8d13 100644 --- a/src/yellow/syntax.rs +++ b/src/yellow/syntax.rs | |||
@@ -1,18 +1,20 @@ | |||
1 | use std::{ | 1 | use std::{ |
2 | fmt, | 2 | fmt, |
3 | sync::Arc, | 3 | sync::Arc, |
4 | ptr | ||
4 | }; | 5 | }; |
5 | 6 | ||
6 | use { | 7 | use { |
7 | TextRange, TextUnit, | 8 | TextRange, TextUnit, |
8 | SyntaxKind::{self, *}, | 9 | SyntaxKind::{self, *}, |
9 | yellow::{Ptr, RedNode, GreenNode}, | 10 | yellow::{RedNode, GreenNode}, |
10 | }; | 11 | }; |
11 | 12 | ||
12 | #[derive(Clone)] | 13 | #[derive(Clone)] |
13 | pub struct SyntaxNode { | 14 | pub struct SyntaxNode { |
14 | pub(crate) root: SyntaxRoot, | 15 | pub(crate) root: SyntaxRoot, |
15 | red: Ptr<RedNode>, | 16 | // guaranteed to be alive bc SyntaxRoot holds a strong ref |
17 | red: ptr::NonNull<RedNode>, | ||
16 | } | 18 | } |
17 | 19 | ||
18 | #[derive(Clone)] | 20 | #[derive(Clone)] |
@@ -29,10 +31,10 @@ pub(crate) struct SyntaxError { | |||
29 | 31 | ||
30 | impl SyntaxNode { | 32 | impl SyntaxNode { |
31 | pub(crate) fn new(root: GreenNode, errors: Vec<SyntaxError>) -> SyntaxNode { | 33 | pub(crate) fn new(root: GreenNode, errors: Vec<SyntaxError>) -> SyntaxNode { |
32 | let root = Arc::new(RedNode::new_root(root)); | 34 | let red = Arc::new(RedNode::new_root(root)); |
33 | let red = Ptr::new(&root); | 35 | let red_weak: ptr::NonNull<RedNode> = (&*red).into(); |
34 | let root = SyntaxRoot { red: root, errors: Arc::new(errors) }; | 36 | let root = SyntaxRoot { red, errors: Arc::new(errors) }; |
35 | SyntaxNode { root, red } | 37 | SyntaxNode { root, red: red_weak } |
36 | } | 38 | } |
37 | 39 | ||
38 | pub fn kind(&self) -> SyntaxKind { | 40 | pub fn kind(&self) -> SyntaxKind { |
@@ -58,15 +60,14 @@ impl SyntaxNode { | |||
58 | for i in 0..n_children { | 60 | for i in 0..n_children { |
59 | res.push(SyntaxNode { | 61 | res.push(SyntaxNode { |
60 | root: self.root.clone(), | 62 | root: self.root.clone(), |
61 | red: Ptr::new(&red.nth_child(Ptr::clone(&self.red), i)), | 63 | red: (&*red.nth_child(i)).into(), |
62 | }); | 64 | }); |
63 | } | 65 | } |
64 | res | 66 | res |
65 | } | 67 | } |
66 | 68 | ||
67 | fn red(&self) -> &RedNode { | 69 | fn red(&self) -> &RedNode { |
68 | // Safe b/c root ptr keeps red alive | 70 | unsafe { self.red.as_ref() } |
69 | unsafe { self.red.get() } | ||
70 | } | 71 | } |
71 | } | 72 | } |
72 | 73 | ||