From 67932548cdfda2bf5179a337034760627914196a Mon Sep 17 00:00:00 2001 From: Clement Rey Date: Fri, 22 Sep 2023 16:36:53 +0200 Subject: [PATCH] fix all zombie issues --- crates/re_arrow_store/src/store.rs | 3 --- crates/re_arrow_store/src/store_gc.rs | 3 --- crates/re_arrow_store/src/store_write.rs | 2 -- crates/re_components/src/load_file.rs | 2 +- crates/re_log_types/src/data_cell.rs | 3 --- crates/re_renderer/src/point_cloud_builder.rs | 2 -- crates/re_renderer/src/renderer/lines.rs | 4 ++-- crates/re_renderer/src/renderer/point_cloud.rs | 2 +- crates/re_sdk/src/recording_stream.rs | 2 +- .../src/contexts/transform_context.rs | 4 ++-- crates/re_space_view_spatial/src/parts/assets.rs | 4 ++-- crates/re_space_view_spatial/src/ui_2d.rs | 2 +- .../definitions/rerun/archetypes/points2d.fbs | 3 --- crates/re_types/source_hash.txt | 2 +- crates/re_types_builder/src/codegen/cpp/mod.rs | 5 ----- docs/code-examples/clear_recursive.py | 2 +- docs/code-examples/clear_simple.py | 2 +- docs/code-examples/segmentation_image_simple.py | 3 +-- docs/content/reference/viewer/viewport.md | 2 -- examples/python/arkit_scenes/main.py | 12 ++++++------ rerun_py/docs/gen_common_index.py | 2 -- rerun_py/rerun_sdk/rerun/_baseclasses.py | 2 +- rerun_py/rerun_sdk/rerun/archetypes/clear.py | 4 ++-- .../rerun_sdk/rerun/archetypes/segmentation_image.py | 3 +-- tests/roundtrips.py | 12 ++++++------ tests/rust/test_api/src/main.rs | 2 +- 26 files changed, 31 insertions(+), 58 deletions(-) diff --git a/crates/re_arrow_store/src/store.rs b/crates/re_arrow_store/src/store.rs index 945ffe431bfc..baf55f9bbcf5 100644 --- a/crates/re_arrow_store/src/store.rs +++ b/crates/re_arrow_store/src/store.rs @@ -371,8 +371,6 @@ fn datastore_internal_repr() { /// ```text /// cargo test -p re_arrow_store -- --nocapture datastore_internal_repr /// ``` -// -// TODO(#1524): inline visualization once it's back to a manageable state #[derive(Debug, Clone)] pub struct IndexedTable { /// The timeline this table operates in, for debugging purposes. @@ -539,7 +537,6 @@ impl Default for IndexedBucketInner { /// cargo test -p re_arrow_store -- --nocapture datastore_internal_repr /// ``` // -// TODO(#1524): inline visualization once it's back to a manageable state // TODO(#1807): timeless should be row-id ordered too then #[derive(Debug, Clone)] pub struct PersistentIndexedTable { diff --git a/crates/re_arrow_store/src/store_gc.rs b/crates/re_arrow_store/src/store_gc.rs index c70143ac2228..6558f8a6fc24 100644 --- a/crates/re_arrow_store/src/store_gc.rs +++ b/crates/re_arrow_store/src/store_gc.rs @@ -111,9 +111,6 @@ impl DataStore { // datastructures should be able to purge themselves based solely off of // [`DataStore::oldest_time_per_timeline`]. // - // TODO(#1803): The GC should be aware of latest-at semantics and make sure they are upheld - // when purging data. - // // TODO(#1823): Workload specific optimizations. pub fn gc(&mut self, options: GarbageCollectionOptions) -> (Deleted, DataStoreStats) { re_tracing::profile_function!(); diff --git a/crates/re_arrow_store/src/store_write.rs b/crates/re_arrow_store/src/store_write.rs index 102a0ff6009a..6d539b09d1a5 100644 --- a/crates/re_arrow_store/src/store_write.rs +++ b/crates/re_arrow_store/src/store_write.rs @@ -545,8 +545,6 @@ impl IndexedBucket { /// ```text /// cargo test -p re_arrow_store -- --nocapture datastore_internal_repr /// ``` - // - // TODO(#1524): inline visualization once it's back to a manageable state fn split(&self) -> Option<(TimeInt, Self)> { let Self { timeline, diff --git a/crates/re_components/src/load_file.rs b/crates/re_components/src/load_file.rs index ff98dffcf5a1..c874ae29edf6 100644 --- a/crates/re_components/src/load_file.rs +++ b/crates/re_components/src/load_file.rs @@ -143,7 +143,7 @@ pub fn data_cells_from_mesh_file_contents( bytes: Vec, format: crate::MeshFormat, ) -> Result, FromFileError> { - // TODO(#2788): mesh indicator + // TODO(#3354): mesh indicator let mesh = crate::EncodedMesh3D { format, bytes: bytes.into(), diff --git a/crates/re_log_types/src/data_cell.rs b/crates/re_log_types/src/data_cell.rs index 2e61ab8f7055..b13d9a708dd8 100644 --- a/crates/re_log_types/src/data_cell.rs +++ b/crates/re_log_types/src/data_cell.rs @@ -276,9 +276,6 @@ impl DataCell { // --- /// Builds an empty `DataCell` from a native component type. - // - // TODO(#1595): do keep in mind there's a future not too far away where components become a - // `(component, type)` tuple kinda thing. #[inline] pub fn from_native_empty() -> Self { Self::from_arrow_empty(C::name(), C::arrow_field().data_type) diff --git a/crates/re_renderer/src/point_cloud_builder.rs b/crates/re_renderer/src/point_cloud_builder.rs index c8b424b6f3a1..9c39a6d1e68b 100644 --- a/crates/re_renderer/src/point_cloud_builder.rs +++ b/crates/re_renderer/src/point_cloud_builder.rs @@ -155,8 +155,6 @@ impl<'a> PointCloudBatchBuilder<'a> { /// Will add all positions. /// Missing radii will default to `Size::AUTO`. /// Missing colors will default to white. - /// - /// TODO(#957): Clamps number of points to the allowed per-builder maximum. #[inline] pub fn add_points( mut self, diff --git a/crates/re_renderer/src/renderer/lines.rs b/crates/re_renderer/src/renderer/lines.rs index f5ef9ef0b4cc..0517b3ebc987 100644 --- a/crates/re_renderer/src/renderer/lines.rs +++ b/crates/re_renderer/src/renderer/lines.rs @@ -356,13 +356,13 @@ const LINE_STRIP_TEXTURE_SIZE: u32 = 256; // 256 x 256 x vec2 == 0.5MiB, 65 impl LineDrawData { /// Total maximum number of line vertices per [`LineDrawData`]. /// - /// TODO(#957): Get rid of this limit!. + /// TODO(#3076): Get rid of this limit!. pub const MAX_NUM_VERTICES: usize = (POSITION_TEXTURE_SIZE * POSITION_TEXTURE_SIZE - 2) as usize; // Subtract sentinels /// Total maximum number of line strips per [`LineDrawData`]. /// - /// TODO(#957): Get rid of this limit!. + /// TODO(#3076): Get rid of this limit!. pub const MAX_NUM_STRIPS: usize = (LINE_STRIP_TEXTURE_SIZE * LINE_STRIP_TEXTURE_SIZE) as usize; /// Transforms and uploads line strip data to be consumed by gpu. diff --git a/crates/re_renderer/src/renderer/point_cloud.rs b/crates/re_renderer/src/renderer/point_cloud.rs index 25247f89b167..b75c114871d6 100644 --- a/crates/re_renderer/src/renderer/point_cloud.rs +++ b/crates/re_renderer/src/renderer/point_cloud.rs @@ -169,7 +169,7 @@ const DATA_TEXTURE_SIZE: u32 = 2048; // 2ki x 2ki = 4 Mi = 80 MiB impl PointCloudDrawData { /// Maximum number of vertices per [`PointCloudDrawData`]. /// - /// TODO(#957): Get rid of this limit!. + /// TODO(#3076): Get rid of this limit!. pub const MAX_NUM_POINTS: usize = (DATA_TEXTURE_SIZE * DATA_TEXTURE_SIZE) as usize; /// Transforms and uploads point cloud data to be consumed by gpu. diff --git a/crates/re_sdk/src/recording_stream.rs b/crates/re_sdk/src/recording_stream.rs index 4008286d52c9..2e84a9f174a1 100644 --- a/crates/re_sdk/src/recording_stream.rs +++ b/crates/re_sdk/src/recording_stream.rs @@ -711,7 +711,7 @@ impl RecordingStream { ) }); - // TODO(#1629): unsplit splats once new data cells are in + // TODO(#1893): unsplit splats once new data cells are in let splatted = (!splatted.is_empty()).then(|| { splatted.push(DataCell::from_native([InstanceKey::SPLAT])); DataRow::from_cells(RowId::random(), timepoint, ent_path, 1, splatted) diff --git a/crates/re_space_view_spatial/src/contexts/transform_context.rs b/crates/re_space_view_spatial/src/contexts/transform_context.rs index b8338b69af0b..6ea1dde8cf7c 100644 --- a/crates/re_space_view_spatial/src/contexts/transform_context.rs +++ b/crates/re_space_view_spatial/src/contexts/transform_context.rs @@ -130,7 +130,7 @@ impl ViewContextSystem for TransformContext { ¤t_tree.path, &entity_db.data_store, &time_query, - // TODO(#1988): See comment in transform_at. This is a workaround for precision issues + // TODO(#1025): See comment in transform_at. This is a workaround for precision issues // and the fact that there is no meaningful image plane distance for 3D->2D views. |_| 500.0, &mut encountered_pinhole, @@ -330,7 +330,7 @@ fn transform_at( // Above calculation is nice for a certain kind of visualizing a projected image plane, // but the image plane distance is arbitrary and there might be other, better visualizations! - // TODO(#1988): + // TODO(#1025): // As such we don't ever want to invert this matrix! // However, currently our 2D views require do to exactly that since we're forced to // build a relationship between the 2D plane and the 3D world, when actually the 2D plane diff --git a/crates/re_space_view_spatial/src/parts/assets.rs b/crates/re_space_view_spatial/src/parts/assets.rs index 349f703897bd..165a63170d40 100644 --- a/crates/re_space_view_spatial/src/parts/assets.rs +++ b/crates/re_space_view_spatial/src/parts/assets.rs @@ -89,12 +89,12 @@ impl ViewPartSystem for Asset3DPart { std::iter::once(Mesh3D::name()).collect() } - // TODO(#2788): use this instead + // TODO(#3354): use this instead // fn archetype(&self) -> Vec { // Mesh3D::required_components().to_vec() // } - // TODO(#2788): use this instead + // TODO(#3354): use this instead // fn heuristic_filter( // &self, // _store: &re_arrow_store::DataStore, diff --git a/crates/re_space_view_spatial/src/ui_2d.rs b/crates/re_space_view_spatial/src/ui_2d.rs index b49c40dc0a67..d7ea4e537ca7 100644 --- a/crates/re_space_view_spatial/src/ui_2d.rs +++ b/crates/re_space_view_spatial/src/ui_2d.rs @@ -398,7 +398,7 @@ fn setup_target_config( gpu_bridge::viewport_resolution_in_pixels(egui_painter.clip_rect(), pixels_from_points); anyhow::ensure!(resolution_in_pixel[0] > 0 && resolution_in_pixel[1] > 0); - // TODO(#1988): + // TODO(#1025): // The camera setup is done in a way that works well with the way we inverse pinhole camera transformations right now. // This has a lot of issues though, mainly because we pretend that the 2D plane has a defined depth. // * very bad depth precision as we limit the depth range from 0 to focal_length_in_pixels diff --git a/crates/re_types/definitions/rerun/archetypes/points2d.fbs b/crates/re_types/definitions/rerun/archetypes/points2d.fbs index acc40647e298..55460e898c6a 100644 --- a/crates/re_types/definitions/rerun/archetypes/points2d.fbs +++ b/crates/re_types/definitions/rerun/archetypes/points2d.fbs @@ -7,11 +7,8 @@ namespace rerun.archetypes; // --- -// TODO(#2371): archetype IDL definitions must always be tables -// TODO(#2372): archetype IDL definitions must refer to objects of kind component // TODO(#2373): `attr.rerun.component_required` implies `required` // TODO(#2427): distinguish optional vs. recommended in language backends -// TODO(#2521): always derive debug & clone for rust backend /// A 2D point cloud with positions and optional colors, radii, labels, etc. /// diff --git a/crates/re_types/source_hash.txt b/crates/re_types/source_hash.txt index d4bcb2884b55..dcf61ade5376 100644 --- a/crates/re_types/source_hash.txt +++ b/crates/re_types/source_hash.txt @@ -1,4 +1,4 @@ # This is a sha256 hash for all direct and indirect dependencies of this crate's build script. # It can be safely removed at anytime to force the build script to run again. # Check out build.rs to see how it's computed. -6f8f25bbe0900f90a47718d6993eaf116ba1e7ac8b59762162bbfe633b5a7e50 +d865d0a05e7038d00db910891e4ab659951d7c0d15a1e5034be6422892b2d42a diff --git a/crates/re_types_builder/src/codegen/cpp/mod.rs b/crates/re_types_builder/src/codegen/cpp/mod.rs index 7eb0e3e5cdb2..4e5b80373487 100644 --- a/crates/re_types_builder/src/codegen/cpp/mod.rs +++ b/crates/re_types_builder/src/codegen/cpp/mod.rs @@ -38,7 +38,6 @@ const DOC_COMMENT_SUFFIX_TOKEN: &str = "DOC_COMMENT_SUFFIX_TOKEN"; const ANGLE_BRACKET_LEFT_TOKEN: &str = "SYS_INCLUDE_PATH_PREFIX_TOKEN"; const ANGLE_BRACKET_RIGHT_TOKEN: &str = "SYS_INCLUDE_PATH_SUFFIX_TOKEN"; const HEADER_EXTENSION_TOKEN: &str = "HEADER_EXTENSION_TOKEN"; -const TODO_TOKEN: &str = "TODO_TOKEN"; fn quote_comment(text: &str) -> TokenStream { quote! { #NORMAL_COMMENT_PREFIX_TOKEN #text #NORMAL_COMMENT_SUFFIX_TOKEN } @@ -67,10 +66,6 @@ fn string_from_token_stream(token_stream: &TokenStream, source_path: Option<&Utf .replace(&format!("\" {DOC_COMMENT_SUFFIX_TOKEN:?}"), "\n") .replace(&format!("{ANGLE_BRACKET_LEFT_TOKEN:?} \""), "<") .replace(&format!("\" {ANGLE_BRACKET_RIGHT_TOKEN:?}"), ">") - .replace( - &format!("{TODO_TOKEN:?}"), - "\n// TODO(#2647): code-gen for C++\n", - ) .replace("< ", "<") .replace(" >", ">") .replace(" ::", "::"); diff --git a/docs/code-examples/clear_recursive.py b/docs/code-examples/clear_recursive.py index 44410313cb32..aa2eb30e1347 100644 --- a/docs/code-examples/clear_recursive.py +++ b/docs/code-examples/clear_recursive.py @@ -14,5 +14,5 @@ rr2.log(f"arrows/{i}", rr2.Arrows3D(vector, origins=origin, colors=color)) # Now clear all of them at once. -# TODO(#3268): `rr2.Clear.recursive()` +# TODO(cmc): `rr2.Clear.recursive()` rr2.log("arrows", rr2.Clear(True)) diff --git a/docs/code-examples/clear_simple.py b/docs/code-examples/clear_simple.py index 3a654ff2b0d7..cfe22bed2252 100644 --- a/docs/code-examples/clear_simple.py +++ b/docs/code-examples/clear_simple.py @@ -15,5 +15,5 @@ # Now clear them, one by one on each tick. for i in range(len(vectors)): - # TODO(#3268): `rr2.Clear.flat()` + # TODO(cmc): `rr2.Clear.flat()` rr2.log(f"arrows/{i}", rr2.Clear(False)) diff --git a/docs/code-examples/segmentation_image_simple.py b/docs/code-examples/segmentation_image_simple.py index fb744f1649c3..82bbc2ed0c4e 100644 --- a/docs/code-examples/segmentation_image_simple.py +++ b/docs/code-examples/segmentation_image_simple.py @@ -14,5 +14,4 @@ # Assign a label and color to each class rr2.log("/", rr2.AnnotationContext([(1, "red", (255, 0, 0)), (2, "green", (0, 255, 0))])) -# TODO(#2792): SegmentationImage archetype -rr.log_segmentation_image("image", np.array(image)) +rr2.log("image", rr2.SegmentationImage(image)) diff --git a/docs/content/reference/viewer/viewport.md b/docs/content/reference/viewer/viewport.md index bb5711392475..890b1a154683 100644 --- a/docs/content/reference/viewer/viewport.md +++ b/docs/content/reference/viewer/viewport.md @@ -39,6 +39,4 @@ Rerun distinguishes various categories of Space Views: Which category is used is determined upon creation of a Space View. -[TODO(#1164)](https://github.com/rerun-io/rerun/issues/1164): Allow configuring the category of a space view after its creation. - The kind of Space View determines which Entities it can display, how it displays them and the way they can be interacted with. diff --git a/examples/python/arkit_scenes/main.py b/examples/python/arkit_scenes/main.py index edd6475e0578..db9343bc2965 100755 --- a/examples/python/arkit_scenes/main.py +++ b/examples/python/arkit_scenes/main.py @@ -43,7 +43,7 @@ def log_annotated_bboxes(annotation: dict[str, Any]) -> tuple[npt.NDArray[np.flo Logs annotated oriented bounding boxes to Rerun. We currently calculate and return the 3D bounding boxes keypoints, labels, and colors for each object to log them in - each camera frame TODO(#1581): once resolved this can be removed. + each camera frame TODO(#3412): once resolved this can be removed. annotation json file | |-- label: object name of bounding box @@ -55,7 +55,7 @@ def log_annotated_bboxes(annotation: dict[str, Any]) -> tuple[npt.NDArray[np.flo bbox_labels = [] num_objects = len(annotation["data"]) # Generate a color per object that can be reused across both 3D obb and their 2D projections - # TODO(#1581, #1728): once resolved this can be removed + # TODO(#3412, #1728): once resolved this can be removed color_positions = np.linspace(0, 1, num_objects) colormap = plt.colormaps["viridis"] colors = [colormap(pos) for pos in color_positions] @@ -93,7 +93,7 @@ def compute_box_3d( """ Given obb compute 3d keypoints of the box. - TODO(#1581): once resolved this can be removed + TODO(#3412): once resolved this can be removed """ length, height, width = half_size.tolist() center = np.reshape(transform, (-1, 3)) @@ -123,7 +123,7 @@ def log_line_segments(entity_path: str, bboxes_2d_filtered: npt.NDArray[np.float |/ |/ 1 -------- 0 - TODO(#1581): once resolved this can be removed + TODO(#3412): once resolved this can be removed :param bboxes_2d_filtered: A numpy array of shape (8, 2), representing the filtered 2D keypoints of the 3D bounding boxes. @@ -175,7 +175,7 @@ def project_3d_bboxes_to_2d_keypoints( """ Returns 2D keypoints of the 3D bounding box in the camera view. - TODO(#1581): once resolved this can be removed + TODO(#3412): once resolved this can be removed Args: bboxes_3d: (nObjects, 8, 3) containing the 3D bounding box keypoints in world frame. camera_from_world: Tuple containing the camera translation and rotation_quaternion in world frame. @@ -242,7 +242,7 @@ def log_camera( intrinsic = np.array([[fx, 0, cx], [0, fy, cy], [0, 0, 1]]) camera_from_world = poses_from_traj[frame_id] - # TODO(#1581): once resolved this can be removed + # TODO(#3412): once resolved this can be removed # Project 3D bounding boxes into 2D image bboxes_2d = project_3d_bboxes_to_2d_keypoints(bboxes, camera_from_world, intrinsic, img_width=w, img_height=h) # clear previous centroid labels diff --git a/rerun_py/docs/gen_common_index.py b/rerun_py/docs/gen_common_index.py index 5035489b8d8b..bdd110891528 100644 --- a/rerun_py/docs/gen_common_index.py +++ b/rerun_py/docs/gen_common_index.py @@ -182,8 +182,6 @@ def make_slug(s: str) -> str: with mkdocs_gen_files.open(index_path, "w") as index_file: - # TODO(#1161): add links to our high-level docs! - # Hide the TOC for the index since it's identical to the left nav-bar index_file.write( """--- diff --git a/rerun_py/rerun_sdk/rerun/_baseclasses.py b/rerun_py/rerun_sdk/rerun/_baseclasses.py index 7ae4e52f0095..a7f095f79c2d 100644 --- a/rerun_py/rerun_sdk/rerun/_baseclasses.py +++ b/rerun_py/rerun_sdk/rerun/_baseclasses.py @@ -57,7 +57,7 @@ def as_component_batches(self) -> Iterable[ComponentBatchLike]: for fld in fields(type(self)): if "component" in fld.metadata: comp = getattr(self, fld.name) - # TODO(#3341): Depending on what we decide + # TODO(#3381): Depending on what we decide # to do with optional components, we may need to make this instead call `_empty_pa_array` if comp is not None: yield comp diff --git a/rerun_py/rerun_sdk/rerun/archetypes/clear.py b/rerun_py/rerun_sdk/rerun/archetypes/clear.py index ac2de22364e5..ffb122805fea 100644 --- a/rerun_py/rerun_sdk/rerun/archetypes/clear.py +++ b/rerun_py/rerun_sdk/rerun/archetypes/clear.py @@ -40,7 +40,7 @@ class Clear(Archetype): # Now clear them, one by one on each tick. for i in range(len(vectors)): - # TODO(#3268): `rr2.Clear.flat()` + # TODO(cmc): `rr2.Clear.flat()` rr2.log(f"arrows/{i}", rr2.Clear(False)) ``` @@ -61,7 +61,7 @@ class Clear(Archetype): rr2.log(f"arrows/{i}", rr2.Arrows3D(vector, origins=origin, colors=color)) # Now clear all of them at once. - # TODO(#3268): `rr2.Clear.recursive()` + # TODO(cmc): `rr2.Clear.recursive()` rr2.log("arrows", rr2.Clear(True)) ``` """ diff --git a/rerun_py/rerun_sdk/rerun/archetypes/segmentation_image.py b/rerun_py/rerun_sdk/rerun/archetypes/segmentation_image.py index eda0c8438055..5b53d70fa38e 100644 --- a/rerun_py/rerun_sdk/rerun/archetypes/segmentation_image.py +++ b/rerun_py/rerun_sdk/rerun/archetypes/segmentation_image.py @@ -45,8 +45,7 @@ class SegmentationImage(SegmentationImageExt, Archetype): # Assign a label and color to each class rr2.log("/", rr2.AnnotationContext([(1, "red", (255, 0, 0)), (2, "green", (0, 255, 0))])) - # TODO(#2792): SegmentationImage archetype - rr.log_segmentation_image("image", np.array(image)) + rr2.log("image", rr2.SegmentationImage(image)) ``` """ diff --git a/tests/roundtrips.py b/tests/roundtrips.py index 79e5ead0f038..d34781671bf3 100755 --- a/tests/roundtrips.py +++ b/tests/roundtrips.py @@ -23,15 +23,15 @@ # TODO(#3207): implement missing cpp roundtrips opt_out = { "clear": ["cpp", "py", "rust"], # Don't need it, API example roundtrips cover it all - "time_series_scalar": ["cpp", "py", "rust"], # Don't need it, API example roundtrips cover it all - "depth_image": ["cpp"], # TODO(#2792) - "image": ["cpp"], # TODO(#2792) + "depth_image": ["cpp"], # TODO(#3380) + "image": ["cpp"], # TODO(#3380) "mesh3d": ["cpp", "py", "rust"], # Don't need it, API example roundtrips cover it all - "segmentation_image": ["cpp"], # TODO(#2792) - "tensor": ["cpp"], # TODO(#2792) + "segmentation_image": ["cpp"], # TODO(#3380) + "tensor": ["cpp"], # TODO(#3380) + "time_series_scalar": ["cpp", "py", "rust"], # Don't need it, API example roundtrips cover it all "transform3d": [ "cpp" - ], # TODO(#2792) - `tests/roundtrips.py --release transform3d` fails on mac (but works without --release) + ], # TODO(#3380) - `tests/roundtrips.py --release transform3d` fails on mac (but works without --release) } diff --git a/tests/rust/test_api/src/main.rs b/tests/rust/test_api/src/main.rs index 94bbe25983f5..dfb73267a742 100644 --- a/tests/rust/test_api/src/main.rs +++ b/tests/rust/test_api/src/main.rs @@ -196,7 +196,7 @@ fn test_rects(rec: &RecordingStream) -> anyhow::Result<()> { rec.set_time_seconds("sim_time", 3f64); rec.log( "rects_test/rects", - // TODO(#3279): Should be &Boxes2D::empty() + // TODO(#3381): Should be &Boxes2D::empty() &Boxes2D::from_half_sizes(std::iter::empty::()), )?;