forked from igorbologovv/lodtree
-
-
Notifications
You must be signed in to change notification settings - Fork 0
/
rayon.rs
113 lines (86 loc) · 4.03 KB
/
rayon.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
/* Generic tree structures for storage of spatial data.
Copyright (C) 2023 Alexander Pyattaev
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
/*use spatialtree::*;
use rayon::prelude::*;
struct Chunk {
// data to store in the chunk, for EG storing voxel or heighmap data
// if you don't use editing, storing this data isn't needed, and only storing the mesh would be enough
data: [f32; 4096], // this amount of data actually makes it slower. To see the true octree speed, replace this with [f32; 0]
}
impl Chunk {
// this does a cheap init so it can safely be put inside the vec
fn new(_position: OctVec) -> Self {
Self { data: [0.0; 4096] }
}
// pretend this inits the data with some expensive procedural generation
fn expensive_init(&mut self, _position: OctVec) {
self.data = [1.0; 4096];
// emulate a 1ms time to do things
// we can't use sleep because thats 15ms on windows min
let start = std::time::Instant::now();
while start.elapsed() < std::time::Duration::from_millis(1) {}
}
// and pretend this makes chunks visible/invisible
fn set_visible(&mut self, _visibility: bool) {}
// and pretend this drops anything for when a chunk is permanently deleted
fn cleanup(&mut self) {}
}
fn main() {
// create an octree
let mut tree = OctTree::<Chunk, OctVec>::with_capacity(512, 512);
// the game loop that runs for 42 iterations
for _ in 0..42 {
let start_time = std::time::Instant::now();
// get the pending updates
if tree.prepare_update(
&[OctVec::new(4096, 4096, 4096, 32)], // target position in the tree
2, // the amount of detail
&mut |position_in_tree| Chunk::new(position_in_tree), // and how we should make a new tree inside the function here. This should be done quickly
) {
let duration = start_time.elapsed().as_micros();
println!(
"Took {} microseconds to get the tree update ready",
duration
);
//TODO: rewrite this to use chunks_add callback which modifies captured Vec and pushes chunk references
// in there for thread pool to consume later.
// if there was an update, we need to first generate new chunks with expensive_init
tree.get_chunks_to_add_slice_mut().par_iter_mut().for_each(
|ToAddContainer {
position, chunk, ..
}| {
// and run expensive init
chunk.expensive_init(*position);
},
);
let start_time = std::time::Instant::now();
// and don't forget to actually run the update
tree.do_update();
// now we probably want to truly clean up the chunks that are going to be deleted from memory
for chunk in tree.iter_chunks_to_delete_mut() {
chunk.cleanup();
}
// and actually clean them up
tree.complete_update();
let duration = start_time.elapsed().as_micros();
println!("Took {} microseconds to execute the update", duration);
}
let duration = start_time.elapsed().as_micros();
println!("Took {} microseconds to do the entire update", duration);
// and print some data about the run
println!("Num chunks in the tree: {}", tree.get_num_chunks());
}
}
*/
fn main() {}