Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

minimize maxwellian vector capacity somewhat #898

Open
wants to merge 4 commits into
base: master
Choose a base branch
from

Conversation

PhilipDeegan
Copy link
Member

@PhilipDeegan PhilipDeegan commented Oct 3, 2024

to be considered also https://github.com/PhilipDeegan/PHARE/blob/mkn/src/core/vector.hpp#L337-L370

Summary by CodeRabbit

  • New Features

    • Added methods for refining and coarsening box dimensions based on a refinement ratio.
    • Introduced a new test class for 2D Maxwellian particle initialization.
  • Improvements

    • Enhanced particle loading efficiency with updated logic for estimating expected sizes.
    • Updated constructors and method signatures for better usability and flexibility in particle management.
  • Refactor

    • Streamlined type definitions and method implementations for clarity and maintainability.
    • Enhanced the organization and clarity of code related to particle refinement and management.

Copy link

coderabbitai bot commented Oct 3, 2024

📝 Walkthrough
📝 Walkthrough

Walkthrough

The changes in this pull request introduce modifications across several files, primarily focusing on the particles_data_split.hpp, test_particle_data_refine_tag_strategy.hpp, test_particledata_refine_basic_hierarchy.hpp, and test_split.cpp. Key updates include the addition of new type aliases, method signatures, and adjustments to existing methods to improve the handling of particle data. Notably, the loadCell function's return type is changed to void, and a new method for retrieving domain particles is introduced in the TagStrategy class. The overall structure and readability of the code have been enhanced.

Changes

File Change Summary
src/amr/data/particles/refine/particles_data_split.hpp - Added type alias using Particle_t = typename ParticleArray::value_type; in ParticlesRefineOperator.
- Added method static void _reserve(ParticleArray& particles, Fn&& counter).
- Updated refine method signature.
tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp - Updated loadCell method signature from returning ParticleArray<dimension> to void.
- Added method auto domainParticlesForLevel(...) in TagStrategy.
tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp - Updated BasicHierarchy constructor default parameter _ratio from required to default value of 2.
- Added type aliases Splitter_t and ParticlesRefineOperator_t.
- Added method domainParticlesForLevel(int levelNumber).
tests/amr/data/particles/refine/test_split.cpp - Updated type alias from using Splitter to using Splitter_t.
- Added type alias using BasicHierarchy_t = BasicHierarchy<2, 3, ParticlesDataSplitType::interior, 4>;.
- Created test fixture Hierarchy_t and added test case.

Possibly related PRs

  • keep particle overallocation on restore state #897: The changes in this PR involve managing ParticleArray objects, which relates to the handling of particles in the main PR, particularly with the introduction of new methods and type aliases that streamline particle management.

Suggested reviewers

  • nicolasaunai
  • rochSmets

Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

❤️ Share
🪧 Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>, please review it.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai gather interesting stats about this repository and render them as a table. Additionally, render a pie chart showing the language distribution in the codebase.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (Invoked using PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai generate docstrings to generate docstrings for this PR. (Beta)
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Other keywords and placeholders

  • Add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.
  • Add @coderabbitai summary to generate the high-level summary at a specific location in the PR description.
  • Add @coderabbitai anywhere in the PR title to generate the title automatically.

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Caution

Inline review comments failed to post

Actionable comments posted: 1

🧹 Outside diff range and nitpick comments (6)
src/core/data/particles/particle.hpp (1)

42-43: Approve with suggestion: Consider using named constants for size values

The addition of the sizes array and the static assertion improves type safety and helps catch potential issues at compile-time. This is a good practice.

However, to improve maintainability, consider replacing the magic numbers (52, 64, 76) with named constants. This would make the code more self-documenting and easier to update if the structure changes in the future.

Example improvement:

constexpr static std::uint8_t PARTICLE_SIZE_1D = 52;
constexpr static std::uint8_t PARTICLE_SIZE_2D = 64;
constexpr static std::uint8_t PARTICLE_SIZE_3D = 76;

std::array<std::uint8_t, 3> constexpr static sizes = {PARTICLE_SIZE_1D, PARTICLE_SIZE_2D, PARTICLE_SIZE_3D};
src/core/utilities/box/box.hpp (1)

135-141: Approve with suggestions: surface_cell_count() method

The addition of the surface_cell_count() method is a good feature. However, I have a few suggestions to improve it:

  1. Add a runtime assertion to check the assumption that the box is never smaller than 3 in any direction. This will help catch potential misuse.

  2. Consider adding a comment explaining the calculation for better maintainability.

  3. Use std::size_t as the explicit return type instead of auto for clarity.

Here's a suggested implementation incorporating these changes:

NO_DISCARD std::size_t surface_cell_count() const
{
    // Calculate the number of cells on the surface of the box
    // by subtracting the inner volume from the total volume
    auto const shape_ = shape();
    for (std::size_t i = 0; i < dim; ++i) {
        assert(shape_[i] >= 3 && "Box must be at least 3 cells wide in each dimension");
    }
    auto const nested = shape_ - 2;
    return core::product(shape_) - core::product(nested);
}

These changes will improve the robustness and readability of the method.

src/amr/solvers/solver_ppc.hpp (2)

155-170: LGTM! Consider adding a comment explaining the overallocation strategy.

The add_to method effectively manages the capacity of ParticleArray objects in the map while preserving overallocation. This approach can help reduce memory reallocation overhead.

Consider adding a brief comment explaining the rationale behind preserving overallocation, as it might not be immediately obvious to other developers why this approach was chosen over a simple assignment.


225-227: LGTM! Consider extracting the key generation to a separate function.

The use of the new add_to method simplifies the logic and ensures consistent handling of ParticleArray objects. This change improves code maintainability and consistency.

To further improve readability, consider extracting the key generation logic into a separate function:

std::string generateKey(const std::string& patchId, const std::string& popName) {
    return patchId + "_" + popName;
}

// Usage in saveState_:
std::string patchId = std::to_string(state.patch->getGlobalId());
for (auto& pop : state.ions) {
    std::string key = generateKey(patchId, pop.name());
    add_to(tmpDomain, key, pop.domainParticles());
    add_to(patchGhost, key, pop.patchGhostParticles());
}

This refactoring would make the saveState_ method more readable and easier to maintain.

tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (1)

103-118: Consider adding a test to verify particles are loaded within the domain

The 2D initializer lacks a test to ensure that particles are loaded within the correct domain boundaries, similar to the loadsParticlesInTheDomain test in the 1D case. Adding such a test would enhance the reliability of your 2D particle initialization.

src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

189-189: Consider making over_alloc_factor configurable

The over_alloc_factor is currently hardcoded to 0.1 and marked with a TODO comment. To improve flexibility and adhere to clean code principles, consider retrieving this value from a configuration or dictionary instead of hardcoding it.

Would you like assistance in implementing this change?

🛑 Comments failed to post (1)
tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (1)

112-112: ⚠️ Potential issue

Use integer arithmetic for 'outer_cell_count' to avoid type conversion issues

Using std::pow with integer arguments returns a double, which can lead to unintended type conversions or precision loss. It's better to perform this calculation using integer arithmetic.

Apply this diff to correct the calculation:

-auto outer_cell_count = std::pow(50, 2) - std::pow(48, 2);
+auto outer_cell_count = (50 * 50) - (48 * 48);
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

    auto outer_cell_count = (50 * 50) - (48 * 48);

@PhilipDeegan PhilipDeegan force-pushed the min_cap branch 2 times, most recently from aa6ea49 to 03b725b Compare October 3, 2024 11:08
Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 4590a2b and 03b725b.

📒 Files selected for processing (4)
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1 hunks)
  • src/core/data/particles/particle.hpp (3 hunks)
  • src/core/utilities/box/box.hpp (2 hunks)
  • tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (3)
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp
  • src/core/data/particles/particle.hpp
  • src/core/utilities/box/box.hpp
🔇 Additional comments (3)
tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (3)

15-20: LGTM: Improved include statements and namespace organization

The addition of the grid layout test header and the reorganization of namespaces enhance code structure and readability. These changes align well with the introduction of the new 2D test class.


28-28: LGTM: Consistent GridLayoutT type alias

The update to the GridLayoutT type alias in the 1D class maintains consistency with the newly introduced 2D class structure. This change improves code uniformity without altering functionality.


79-101: LGTM: Well-structured 2D Maxwellian particle initializer test class

The new AMaxwellianParticleInitializer2D class is well-structured and consistent with the existing 1D class. This addition enhances test coverage for 2D scenarios.

However, I noticed that the number of particles per cell (600) differs from the 1D case (10000). Could you provide a brief explanation for this difference? It might be helpful to add a comment in the code explaining the reasoning behind these values.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Outside diff range and nitpick comments (2)
tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (1)

79-101: LGTM: Well-structured 2D Maxwellian Particle Initializer test class

The new AMaxwellianParticleInitializer2D class is well-designed and consistent with its 1D counterpart. It correctly uses 2D-specific types and initializations.

One minor suggestion:

Consider adding a brief comment explaining why the nbrParticlesPerCell is set to 600 for 2D, as opposed to 10000 for 1D. This would help clarify the reasoning behind the different particle counts.

src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

181-188: Efficient particle count estimation, but consider refining lambda capture.

The introduction of expected_size calculation is an excellent optimization for memory allocation. It efficiently estimates the number of particles by counting cells above the density cutoff.

Consider refining the lambda capture to be more specific:

std::size_t const expected_size
    = [&ndCellIndices, &densityCutOff_ = this->densityCutOff_, nbrParticlePerCell = this->nbrParticlePerCell_](auto const& _n) {
          std::size_t size = 0;
          for (std::size_t flatCellIdx = 0; flatCellIdx < ndCellIndices.size(); ++flatCellIdx)
              if (_n[flatCellIdx] >= densityCutOff_)
                  ++size;
          return size * nbrParticlePerCell;
      }(n);

This change explicitly captures the required variables, improving code clarity and potentially preventing unintended variable captures.

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 03b725b and 4d37e7e.

📒 Files selected for processing (4)
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1 hunks)
  • src/core/data/particles/particle.hpp (3 hunks)
  • src/core/utilities/box/box.hpp (2 hunks)
  • tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (2 hunks)
🚧 Files skipped from review as they are similar to previous changes (2)
  • src/core/data/particles/particle.hpp
  • src/core/utilities/box/box.hpp
🧰 Additional context used
📓 Path-based instructions (1)
src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

🔇 Additional comments (4)
tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (3)

15-20: LGTM: Improved includes and namespace organization

The addition of the test_gridlayout.hpp header and the introduction of the PHARE::initializer::test_fn::func_1d namespace enhance code organization and provide necessary dependencies for the GridLayout testing.


28-29: LGTM: Updated GridLayoutT type alias

The update to use GridLayoutImplYee for the GridLayoutT type alias ensures consistency with the 2D implementation and aligns with the current grid layout system.


73-77: LGTM: Introduced namespace for 2D test functions

The introduction of the PHARE::initializer::test_fn::func_2d namespace effectively separates 2D test functions from 1D, enhancing code organization and maintainability.

src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

195-195: Excellent use of reserve for performance optimization.

The addition of particles.reserve(allocate_size) before the particle generation loop is a great optimization. This pre-allocation strategy has several benefits:

  1. It reduces the number of potential reallocations during particle insertion.
  2. It improves performance by minimizing memory operations.
  3. It works seamlessly with the previously calculated allocate_size.

This change, combined with the expected_size and allocate_size calculations, should significantly improve the memory management and performance of the particle initialization process.

Comment on lines +103 to +119
TEST_F(AMaxwellianParticleInitializer2D, loadsTheCorrectNbrOfParticles)
{
// vector push back allocation observations
// 100 ppc = 262144 - 250000 = 12144 == 12144 * 64 / 1e6 == .7MB overallocated
// 600 ppc = 2097152 - 1500000 = 597152 * 64 / 1e6 == 38MB overallocated

auto const expectedNbrParticles = nbrParticlesPerCell * product(layout.AMRBox().shape());
initializer->loadParticles(particles, layout);
EXPECT_EQ(expectedNbrParticles, particles.size());
auto outer_cell_count = std::pow(50, 2) - std::pow(48, 2);
EXPECT_EQ(particles.capacity(),
particles.size() + (outer_cell_count * nbrParticlesPerCell * .1));

// new method
// 100 ppc = (1511760 - 1500000) * 64 / 1e6 == 0.12544 overallocated
// 600 ppc = (1511760 - 1500000) * 64 / 1e6 == 0.75264 overallocated
}
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Good test case, but consider refining comments and calculations

The test case for 2D particle loading is thorough and checks important aspects like particle count and capacity. However, there are a few areas for improvement:

  1. The comments about vector push back allocation (lines 105-107 and 117-118) seem to be observations or calculations. Consider moving these to separate documentation or adding them as more detailed code comments to explain their significance.

  2. The capacity calculation on lines 113-114 includes an adjustment for outer cells. It might be helpful to add a brief comment explaining the rationale behind this adjustment.

  3. The "new method" comment on line 116 seems incomplete or out of place. Consider removing it or expanding on its meaning if it's still relevant.

To improve clarity and maintainability, consider extracting the capacity calculation into a separate function with a descriptive name, e.g.:

auto calculateExpectedCapacity(const TestGridLayout<GridLayoutT>& layout, std::uint32_t nbrParticlesPerCell) {
    auto outer_cell_count = std::pow(layout.nbrCells()[0], 2) - std::pow(layout.nbrCells()[0] - 2, 2);
    return nbrParticlesPerCell * product(layout.AMRBox().shape()) + 
           (outer_cell_count * nbrParticlesPerCell * 0.1);
}

Then use this function in the test:

EXPECT_EQ(particles.capacity(), calculateExpectedCapacity(layout, nbrParticlesPerCell));

This would make the test more readable and the calculation more reusable.

@nicolasaunai
Copy link
Member

quick comment : this overallocation touches the allocation made in the particle initializer. How about refined level particle arrays? They probably are allocation by pushback from splitting or something like that?

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 6

🧹 Outside diff range and nitpick comments (8)
.github/workflows/cmake_ubuntu.yml (1)

81-85: Consider impact on Maxwellian vector capacity optimization

While these build configuration changes don't directly address the PR's stated objective of minimizing Maxwellian vector capacity, they may indirectly affect the behavior and performance characteristics related to memory allocation and vector capacity:

  1. The Debug build type typically includes additional runtime checks and may alter memory layout.
  2. The -O3 optimization might lead to aggressive inlining and other optimizations that could mask or exacerbate memory usage patterns.

These factors could potentially interfere with accurately assessing the impact of any changes made to minimize vector capacity.

To ensure that the PR's main objective is met:

  1. Consider separating build configuration changes from functional changes to isolate their effects.
  2. Implement specific tests that measure the Maxwellian vector capacity before and after your changes, ensuring they run in both debug and release configurations.
  3. Add memory profiling to your test suite to quantify the impact of your changes on overall memory usage, particularly in the particle initializer mentioned in the PR comments.

This approach will help validate that the vector capacity optimization is effective across different build configurations and doesn't introduce unintended side effects.

src/core/data/particles/particle.hpp (2)

42-42: Consider adding a comment and using a larger integer type for sizes.

The purpose of the sizes array is not immediately clear. Consider adding a comment to explain its usage. Additionally, std::uint8_t might be limiting if the sizes grow larger than 255 bytes in the future. Consider using std::uint16_t or std::size_t for better future-proofing.

-    std::array<std::uint8_t, 3> constexpr static sizes = {52, 64, 76};
+    // Total size in bytes of the Particle struct for dimensions 1, 2, and 3
+    std::array<std::size_t, 3> constexpr static sizes = {52, 64, 76};

Line range hint 1-153: Consider additional improvements for overall code quality.

While the code is generally well-structured, consider the following improvements:

  1. Add more comprehensive documentation, especially for template parameters and member functions.
  2. Consider grouping related functionality (e.g., particle operations) into separate namespaces or classes for better organization.
  3. Use [[nodiscard]] attribute instead of NO_DISCARD macro for better standard compliance.
  4. Consider adding unit tests for the Particle struct and related functions if not already present.
src/core/utilities/box/box.hpp (1)

135-141: Good implementation, but consider some improvements.

The surface_cell_count() function is a useful addition. However, there are a few points to consider:

  1. The assumption that the box is never smaller than 3 in any direction is not enforced. Consider adding a runtime check to ensure this assumption holds.

  2. The comment on line 139 could be more descriptive. Explain why you're subtracting 2 from each dimension (i.e., to exclude the inner volume).

  3. If this function is performance-critical, you might want to consider caching the result or using a more direct calculation method.

Consider updating the implementation like this:

NO_DISCARD auto surface_cell_count() const
{
    // Ensure box is never smaller than 3 in any direction
    for (size_t i = 0; i < dim; ++i) {
        assert(upper[i] - lower[i] + 1 >= 3 && "Box must be at least 3 cells wide in each dimension");
    }
    
    auto const shape_ = shape();
    // Subtract 2 from each dimension to get the inner volume
    auto const inner_shape = shape_ - 2;
    return core::product(shape_) - core::product(inner_shape);
}

This implementation adds a runtime check for the minimum box size and clarifies the purpose of subtracting 2 from each dimension.

tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (2)

90-90: Suggestion: Replace Magic Number with Named Constant

Using a named constant instead of the hardcoded value 50 improves readability and ease of maintenance.

Consider defining a constant:

constexpr std::size_t gridSize = 50;

And then initialize layout with:

: layout{gridSize}

112-112: Suggestion: Avoid Hardcoded Values in Capacity Calculation

Replacing hardcoded values with variables enhances flexibility and maintainability.

Use variables derived from layout:

auto gridSize = layout.nbrCells()[0];
auto outer_cell_count = std::pow(gridSize, 2) - std::pow(gridSize - 2, 2);
src/amr/data/particles/refine/particles_data_split.hpp (2)

108-108: Fix typographical errors in comment

The comment has typographical errors: "Finaly" should be "Finally", and "cartesion" should be "Cartesian".


182-190: Confirm the use of scope for incoming_estimate calculation

The code block from lines 182 to 190 introduces a new scope for calculating incoming_estimate. Ensure that this additional scope is necessary and that variables defined within are appropriately managed.

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 4d37e7e and a62c4ca.

📒 Files selected for processing (6)
  • .github/workflows/cmake_ubuntu.yml (1 hunks)
  • src/amr/data/particles/refine/particles_data_split.hpp (6 hunks)
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (4 hunks)
  • src/core/data/particles/particle.hpp (3 hunks)
  • src/core/utilities/box/box.hpp (2 hunks)
  • tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (2 hunks)
🧰 Additional context used
📓 Path-based instructions (4)
src/amr/data/particles/refine/particles_data_split.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/core/data/particles/particle.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/core/utilities/box/box.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

🔇 Additional comments (15)
.github/workflows/cmake_ubuntu.yml (1)

81-85: Verify impact on CI/CD pipeline performance and consistency

The changes to the build configuration may have significant effects on the CI/CD pipeline:

  1. Build time might be affected due to the conflicting Debug build type and -O3 optimization.
  2. Test execution speed could be unpredictable with this configuration.
  3. Resource usage on CI runners might increase, potentially leading to longer job durations or failures.
  4. The unusual build configuration might make it difficult to reproduce CI issues in local development environments.

To assess the impact of these changes, please run the following checks:

Please review the results to ensure that the changes do not negatively impact the CI/CD pipeline performance or introduce new issues.

src/core/data/particles/particle.hpp (1)

124-126: Improved readability of equality operator template.

The reformatting of the std::enable_if_t condition improves code readability without changing functionality. This is a good change.

src/core/utilities/box/box.hpp (1)

29-29: LGTM: Good addition of type alias.

The new value_type alias improves code readability and follows C++ best practices. It provides a clear representation of the underlying type used in the Box struct.

tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (5)

15-15: Approved: Inclusion of TestGridLayout Header

The addition of test_gridlayout.hpp is appropriate for utilizing TestGridLayout in the tests.


76-78: Approved: Introduction of 2D Test Namespace

The new namespace PHARE::initializer::test_fn::func_2d logically organizes the 2D test functions and maintains consistency with the existing structure.


79-84: Approved: Definition of 2D Particle Initializer Test Class

The AMaxwellianParticleInitializer2D class is correctly defined for testing 2D particle initialization, mirroring the structure of the existing 1D test class.


97-97: Clarify Member Variable Initialization Order

Ensure that member variables are declared in the same order they are initialized to prevent any potential issues with initialization order, especially if dependencies exist between them.

Verify that the declaration order matches the initialization order:

  • Declaration order:

    TestGridLayout<GridLayoutT> layout;
    ParticleArrayT particles{layout.AMRBox()};
  • Initialization order in constructor:

    : layout{gridSize}
    , initializer{...}

Since particles depends on layout, ensure that this order is intentional and functions correctly.


105-119: Duplicate Comment: Refactor Capacity Calculation and Improve Comment Clarity

The previous review comments about refining the capacity calculation and enhancing comment clarity are still applicable.

src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (2)

47-47: Constructor parameter 'over_alloc_factor' added for configurability

The addition of the over_alloc_factor parameter to the constructor enhances the configurability of the class, allowing users to adjust the over-allocation factor as needed.


57-57: Member variable 'over_alloc_factor_' correctly initialized

The member variable over_alloc_factor_ is appropriately initialized in the constructor initializer list with the provided parameter.

src/amr/data/particles/refine/particles_data_split.hpp (5)

5-14: Includes added are appropriate and necessary

The additional include statements provide necessary headers for the new code additions and improve modularity.


56-61: Addition of type alias and constexpr variable enhances code clarity

Defining Particle_t and putParticlesInCoarseBoundary improves code readability and maintainability by simplifying type references and clarifying conditional logic.


142-146: Verify conditional logic in _reserve function

The _reserve function only calls particles.reserve(size) when putParticlesInCoarseBoundary is false. Please confirm that this is the intended behavior, ensuring that reservation is skipped when particles are placed in the coarse boundary.


207-207: Particle position refinement is correctly implemented

Using toFineGrid to compute particleRefinedPos ensures particles are accurately mapped to the fine grid for refinement.


211-211: Initialization and splitting of refined particles

Initializing refinedParticles and applying the split function correctly divides particles for the refinement process.

Comment on lines 81 to 85
-DCMAKE_BUILD_TYPE=Debug -Dasan=OFF \
-DCMAKE_C_COMPILER_LAUNCHER=ccache \
-DCMAKE_CXX_COMPILER_LAUNCHER=ccache \
-DlowResourceTests=ON -DdevMode=ON -Dbench=ON \
-DCMAKE_CXX_FLAGS="-DPHARE_DIAG_DOUBLES=1 " -Dphare_configurator=ON
-DCMAKE_CXX_FLAGS="-O3 -DPHARE_DIAG_DOUBLES=1 " -Dphare_configurator=ON
Copy link

@coderabbitai coderabbitai bot Oct 14, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Conflicting build configuration settings

The changes in the CMake configuration present a potential conflict:

  1. The build type has been changed from RelWithDebInfo to Debug (line 81).
  2. An aggressive optimization flag -O3 has been added to the CMAKE_CXX_FLAGS (line 85).

These settings are contradictory:

  • Debug builds typically include minimal optimizations to facilitate debugging.
  • The -O3 flag enables maximum optimizations, which can interfere with debugging by reordering or optimizing out code.

This combination may lead to unexpected behavior during debugging and potentially mask issues that would be apparent in a true debug build.

Consider one of the following approaches:

  1. If debugging is the priority, remove the -O3 flag and keep the Debug build type.
  2. If performance is crucial even for debugging, consider reverting to RelWithDebInfo build type, which balances optimization and debugging capabilities.

Please clarify the intended goal of these changes to determine the most appropriate configuration.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

RelWithDebInfo includes -DNDEBUG which disables asserts, we want asserts with some optimizations. if we want to debug this build we will change it later

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@PhilipDeegan, understood. Keeping asserts enabled with some optimizations makes sense in your context. I've noted this for future reviews.

(_/)
(•ᴥ•)💡


✏️ Learnings added
Learnt from: PhilipDeegan
PR: PHAREHUB/PHARE#898
File: .github/workflows/cmake_ubuntu.yml:81-85
Timestamp: 2024-10-15T16:27:04.394Z
Learning: In this project, the `RelWithDebInfo` build type includes `-DNDEBUG`, which disables asserts. To keep asserts enabled while having some optimizations, the `Debug` build type is used with the `-O3` optimization flag.

Note: Learnings are effective only in the context of similar code segments. To apply general coding guidelines, please configure review instructions. You can manage existing learnings in the app.

Comment on lines +58 to +62
double weight = 0; // 8
double charge = 0; // 8
std::array<int, dim> iCell = ConstArray<int, dim>(); // 4 * dim
std::array<double, dim> delta = ConstArray<double, dim>(); // 8 * dim
std::array<double, 3> v = ConstArray<double, 3>(); // 8 * 3
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Consider a more maintainable approach for documenting member sizes.

While the added size information is useful, it might become outdated if types change. Consider using sizeof() in static assertions or constexpr variables to automatically calculate and verify sizes. This approach would be more maintainable and less prone to manual errors.

Example implementation:

static_assert(sizeof(weight) == 8, "Unexpected size for weight");
static_assert(sizeof(charge) == 8, "Unexpected size for charge");
static_assert(sizeof(iCell) == 4 * dim, "Unexpected size for iCell");
static_assert(sizeof(delta) == 8 * dim, "Unexpected size for delta");
static_assert(sizeof(v) == 8 * 3, "Unexpected size for v");

Alternatively, you could use Doxygen-style comments for better documentation:

/// @brief Particle weight
/// @details Size: 8 bytes
double weight = 0;

Comment on lines +183 to +188
auto const expected_size = std::accumulate(n, n + ndCellIndices.size(), std::size_t{0},
[&](auto const& sum, auto const& density_value) {
return (density_value > densityCutOff_)
? sum + nbrParticlePerCell_
: sum;
});
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Pass 'sum' by value in the lambda for 'std::accumulate'

In the lambda function used with std::accumulate, the sum parameter should be passed by value instead of by const reference. Passing sum by value is more appropriate since it's an integer type and avoids unnecessary indirection. This aligns with standard practice and can lead to clearer code.

Apply this diff to adjust the lambda parameter:

-auto const expected_size = std::accumulate(n, n + ndCellIndices.size(), std::size_t{0},
-                                           [&](auto const& sum, auto const& density_value) {
+auto const expected_size = std::accumulate(n, n + ndCellIndices.size(), std::size_t{0},
+                                           [&](auto sum, auto const& density_value) {
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
auto const expected_size = std::accumulate(n, n + ndCellIndices.size(), std::size_t{0},
[&](auto const& sum, auto const& density_value) {
return (density_value > densityCutOff_)
? sum + nbrParticlePerCell_
: sum;
});
auto const expected_size = std::accumulate(n, n + ndCellIndices.size(), std::size_t{0},
[&](auto sum, auto const& density_value) {
return (density_value > densityCutOff_)
? sum + nbrParticlePerCell_
: sum;
});

Comment on lines +180 to +182
std::array const particlesArrays{&srcInteriorParticles, &srcGhostParticles};

Copy link

@coderabbitai coderabbitai bot Oct 14, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue

Specify template arguments for std::array declaration

The declaration of particlesArrays is missing the required template arguments for std::array. Please specify the element type and the size to ensure proper compilation.

Apply this diff to correct the declaration:

- std::array const particlesArrays{&srcInteriorParticles, &srcGhostParticles};
+ std::array<ParticleArray const*, 2> const particlesArrays{&srcInteriorParticles, &srcGhostParticles};
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
std::array const particlesArrays{&srcInteriorParticles, &srcGhostParticles};
std::array<ParticleArray const*, 2> const particlesArrays{&srcInteriorParticles, &srcGhostParticles};

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it compiles without the templates, what makes you think it should have the templates?

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@PhilipDeegan, you're right—since the code compiles and class template argument deduction is available from C++17 onwards, your code is valid as is. I apologize for the confusion.

(_/)
(._.)


✏️ Learnings added
Learnt from: PhilipDeegan
PR: PHAREHUB/PHARE#898
File: src/amr/data/particles/refine/particles_data_split.hpp:181-182
Timestamp: 2024-10-16T09:54:01.233Z
Learning: The codebase uses C++17 features such as class template argument deduction, allowing template arguments to be omitted in declarations like `std::array`.

Note: Learnings are effective only in the context of similar code segments. To apply general coding guidelines, please configure review instructions. You can manage existing learnings in the app.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 1

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between a62c4ca and 04b1b69.

📒 Files selected for processing (5)
  • .github/workflows/cmake_ubuntu.yml (1 hunks)
  • src/amr/data/field/refine/electric_field_refiner.hpp (2 hunks)
  • src/amr/data/particles/refine/particles_data_split.hpp (6 hunks)
  • src/amr/utilities/box/amr_box.hpp (2 hunks)
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (4 hunks)
✅ Files skipped from review due to trivial changes (1)
  • src/amr/data/field/refine/electric_field_refiner.hpp
🚧 Files skipped from review as they are similar to previous changes (3)
  • .github/workflows/cmake_ubuntu.yml
  • src/amr/data/particles/refine/particles_data_split.hpp
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp
🧰 Additional context used
📓 Path-based instructions (1)
src/amr/utilities/box/amr_box.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

🔇 Additional comments (1)
src/amr/utilities/box/amr_box.hpp (1)

5-9: LGTM: Include directives are well-organized.

The new include directives are appropriately placed and follow a logical order from core definitions to more specific AMR-related headers. This organization enhances code readability and maintainability.

@PHAREHUB PHAREHUB deleted a comment from coderabbitai bot Oct 15, 2024
@PhilipDeegan
Copy link
Member Author

Todo refinement vector capacity test similar to maxwellian init

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (9)
tests/amr/data/particles/refine/test_split.cpp (2)

41-44: Approve new type alias and struct, suggest documentation.

The addition of BasicHierarchy_t and Hierarchy_t provides a good structure for the new test cases. This setup allows for more specific testing of the hierarchy with the given configuration.

Consider adding brief comments to explain the purpose of these new types and the chosen template parameters for BasicHierarchy_t. This would enhance code readability and maintainability.


46-52: Approve new test case, suggest enhancements.

The new test case checkCapacityPostRefinement is a valuable addition to verify the behavior of domain particles after refinement.

Consider the following enhancements:

  1. Replace assert with ASSERT_* or EXPECT_* macros from Google Test for better error reporting.
  2. Add a brief comment explaining the expected behavior and why these specific values (2800, 4032) are important.
  3. Consider parameterizing the test for different refinement levels or particle counts to increase test coverage.

Example:

TEST_F(Hierarchy_t, checkCapacityPostRefinement)
{
    // Verify that after refinement, the domain particles have the expected size and capacity
    auto domainParticles = domainParticlesForLevel(1);
    ASSERT_EQ(domainParticles.size(), 1);
    EXPECT_EQ(domainParticles[0]->size(), 2800);
    EXPECT_EQ(domainParticles[0]->capacity(), 4032);
}
tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (1)

45-50: Improved efficiency and const-correctness. Consider further optimization.

The changes to the loadCell function are good improvements:

  1. Modifying the passed array in-place instead of returning a new one is more efficient.
  2. Using const for variables that don't change is a good practice.

However, consider further optimization:

void loadCell(ParticleArray<dimension>& particles, std::array<int, 3> const& cellIndices)
{
    float const middle = 0.5f;
    float const delta  = 0.30f;
    // ... rest of the function
}

This change would allow the compiler to optimize the function call by passing the cell indices directly as an array, potentially avoiding some copies.

tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (2)

81-81: LGTM: Improved constructor flexibility with default parameter

The addition of a default value for the _ratio parameter enhances the flexibility of the BasicHierarchy class instantiation. This change maintains backward compatibility while allowing for easier use in scenarios where the default ratio is acceptable.

Consider adding a brief comment explaining the significance of the default ratio value (2) to improve code documentation.


171-175: LGTM: New method added for accessing domain particles

The addition of the domainParticlesForLevel method provides a way to access domain particle information for a specific level, which enhances the functionality of the BasicHierarchy class.

Consider adding documentation for this method, including:

  • A brief description of its purpose
  • The meaning of the levelNumber parameter
  • The type and meaning of the return value

The method appears to be a simple pass-through to tagStrategy_->domainParticlesForLevel. Consider if this design is optimal:

  1. Could this functionality be accessed directly through tagStrategy_?
  2. If this method is necessary, would it be better to return a reference or pointer to tagStrategy_ instead, allowing more flexible use of its methods?

Please review the design choice and ensure it aligns with the overall architecture and usage patterns of the BasicHierarchy class.

src/amr/data/particles/refine/particles_data_split.hpp (4)

27-31: LGTM: New enum class for ParticlesDataSplitType

The addition of the ParticlesDataSplitType enum class is a good improvement. It provides clear categorization for different types of particle data splits and enhances type safety. The use of std::uint8_t as the underlying type is space-efficient.

Consider adding a brief comment explaining the purpose of each enum value for better clarity.


142-147: LGTM: New _reserve method for conditional space reservation

The addition of the _reserve method is a good optimization. It conditionally reserves space in the particle array based on the putParticlesInCoarseBoundary flag, which can improve performance by reducing reallocation. The use of if constexpr for compile-time branching is appropriate.

Consider adding a brief comment explaining the purpose of this method and why reservation is only done for regular refinement.


181-196: LGTM: Improved particle count estimation and space reservation

The addition of particlesArrays and the count_expected lambda function are good improvements. They provide a more accurate estimation of required space, which can enhance performance by reducing reallocation. The use of _reserve with this estimated count is an effective optimization.

Consider adding a brief comment explaining the purpose and logic of the count_expected lambda for better code documentation.


Line range hint 213-265: LGTM: Improved particle refinement logic

The updated particle refinement logic is well-structured and efficient. The use of std::array for refined particles can improve performance compared to dynamic allocation. The conditional compilation for different split types allows for efficient code generation tailored to each case.

Consider adding brief comments explaining the purpose of each major section within this block of code to improve readability for future maintainers.

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 04b1b69 and 2abb9aa.

📒 Files selected for processing (4)
  • src/amr/data/particles/refine/particles_data_split.hpp (6 hunks)
  • tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (3 hunks)
  • tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (4 hunks)
  • tests/amr/data/particles/refine/test_split.cpp (3 hunks)
🧰 Additional context used
📓 Path-based instructions (3)
src/amr/data/particles/refine/particles_data_split.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

🔇 Additional comments (12)
tests/amr/data/particles/refine/test_split.cpp (3)

8-8: LGTM: Necessary include for the new test fixture.

The addition of this include statement is appropriate as it brings in the necessary definitions for the BasicHierarchy used in the new test fixture.


27-27: LGTM: Consistent update of Splitters definition.

The Splitters type definition has been correctly updated to use the new Splitter_t alias, maintaining consistency with the earlier change.


17-17: Approve type alias change and suggest consistency.

The change from Splitter to Splitter_t improves code clarity. It's a good practice to use the _t suffix for type aliases.

Please ensure all occurrences of Splitter are updated to Splitter_t throughout the file for consistency. You can use the following command to check:

✅ Verification successful

Type alias change verified successfully.

All active occurrences of Splitter have been updated to Splitter_t. Only commented code contains Splitter, which is acceptable.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

rg --type cpp 'Splitter(?!_t)' tests/amr/data/particles/refine/test_split.cpp

Length of output: 305


Script:

#!/bin/bash
rg --pcre2 --type cpp 'Splitter(?!_t)' tests/amr/data/particles/refine/test_split.cpp

Length of output: 487

tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (2)

158-158: LGTM: Consistent update with loadCell function change

The call to loadCell has been correctly updated to match its new signature. This change maintains the existing logic while adapting to the more efficient implementation of loadCell.


202-214: New method added. Consider performance and safety improvements.

The new domainParticlesForLevel method provides a useful way to access particle data across all patches in a level. However, there are a few points to consider:

  1. Performance: The use of dynamic_cast in a loop might impact performance if this method is called frequently. Consider if a static_cast would be sufficient here.

  2. Memory safety: Returning raw pointers could lead to ownership and lifetime issues. Consider returning std::vector<std::reference_wrapper<ParticleArray<dimension>>> instead.

  3. Const-correctness: If this method doesn't modify the particle arrays, consider making it const and returning const references.

Here's a suggested improvement:

std::vector<std::reference_wrapper<const ParticleArray<dimension>>> 
domainParticlesForLevel(std::shared_ptr<SAMRAI::hier::PatchHierarchy> const& hierarchy,
                        int levelNumber) const
{
    std::vector<std::reference_wrapper<const ParticleArray<dimension>>> particle_arrays;
    auto level = hierarchy->getPatchLevel(levelNumber);
    for (auto const& patch : *level)
        for (auto const& [name, dataId] : dataToAllocate_)
            if (auto particlesData = static_cast<ParticlesData<ParticleArray<dimension>>*>(
                    patch->getPatchData(dataId)))
                particle_arrays.emplace_back(particlesData->domainParticles);
    return particle_arrays;
}

Could you clarify the intended usage of this method? This would help in determining if these suggestions are appropriate for your use case.

✅ Verification successful

Verification of domainParticlesForLevel Usage

The domainParticlesForLevel method is currently only utilized within test files. Given its limited scope in testing scenarios, the performance and memory safety concerns are minimal. Therefore, the suggested improvements may not be necessary for the current usage.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for uses of domainParticlesForLevel to understand its context
rg "domainParticlesForLevel" --type cpp

Length of output: 602

tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (2)

51-54: LGTM: Improved type aliases enhance code readability

The addition of Splitter_t and ParticlesRefineOperator_t type aliases is a good practice. These aliases encapsulate complex type definitions, making the code more readable and easier to maintain. The use of template parameters also provides flexibility for different use cases.


104-104: LGTM: Simplified refineOperator_ instantiation

The use of the ParticlesRefineOperator_t type alias in the refineOperator_ instantiation improves code readability and consistency. This change simplifies the code without altering its functionality, contributing to better maintainability.

src/amr/data/particles/refine/particles_data_split.hpp (5)

5-6: LGTM: New header inclusions

The addition of new headers for core definitions and utilities is appropriate. This change likely supports new functionality or helps in reducing dependencies in other files.

Also applies to: 8-8


12-14: LGTM: AMR-related header inclusions

The addition of AMR-related utility headers is appropriate for this file's purpose of handling particle data in an AMR context.


56-61: LGTM: Type alias and compile-time constant additions

The new type alias Particle_t improves code readability. The static constexpr boolean putParticlesInCoarseBoundary is a good use of compile-time constants for determining particle placement based on the split type. These additions enhance code clarity and potentially improve performance by allowing compile-time optimizations.


150-152: LGTM: Improved method documentation

The updated comment for the refine_ method provides a clearer and more specific description of its purpose. This improvement in documentation enhances code readability and maintainability.


204-206: LGTM: Optimized splitBox calculation and improved readability

Moving the splitBox calculation outside the loop is a good optimization that avoids redundant calculations. The introduction of the isInDest lambda function improves code readability by encapsulating the condition for checking if a particle is in the destination box. These changes enhance both performance and code clarity.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (6)
tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (1)

171-175: Good addition of domainParticlesForLevel method, but needs documentation

The new domainParticlesForLevel method is a valuable addition to the BasicHierarchy class. It provides a way to access domain particle information for a specific level, which enhances the class's functionality. The method correctly delegates the work to the tagStrategy_ object, maintaining good separation of concerns.

However, the method lacks documentation. To improve code maintainability and make it easier for other developers to understand and use this method, please add a documentation comment.

Consider adding a documentation comment for the domainParticlesForLevel method. For example:

/**
 * @brief Retrieves domain particles for a specific level in the hierarchy.
 * @param levelNumber The level number for which to retrieve domain particles.
 * @return The domain particles for the specified level.
 */
auto domainParticlesForLevel(int levelNumber) const
{
    return tagStrategy_->domainParticlesForLevel(hierarchy_, levelNumber);
}
src/amr/data/particles/refine/particles_data_split.hpp (5)

5-14: LGTM! Consider grouping related headers.

The new header inclusions are appropriate for the functionality of this file. However, consider grouping related headers together for better organization. For example:

// Core headers
#include "core/def.hpp"
#include "phare_core.hpp"
#include "core/def/phare_mpi.hpp"
#include "core/utilities/constants.hpp"

// AMR headers
#include "amr/amr_constants.hpp"
#include "amr/utilities/box/amr_box.hpp"
#include "amr/resources_manager/amr_utils.hpp"
#include "amr/data/particles/particles_data.hpp"

// Local headers
#include "split.hpp"

This grouping can improve readability and make it easier to manage dependencies.


56-61: LGTM! Consider adding a brief comment for putParticlesInCoarseBoundary.

The additions of Particle_t and putParticlesInCoarseBoundary are excellent improvements:

  1. The Particle_t alias enhances code readability and maintainability.
  2. The putParticlesInCoarseBoundary static constexpr boolean encapsulates a complex condition, making the code more self-explanatory.

To further improve clarity, consider adding a brief comment explaining the purpose of putParticlesInCoarseBoundary:

// Determines if particles should be placed in the coarse boundary based on the split type
static constexpr bool putParticlesInCoarseBoundary = ...

This comment would provide context for future maintainers.


142-147: LGTM! Consider adding a brief comment explaining the purpose of _reserve.

The new _reserve method is a great addition:

  1. It improves performance by reserving memory for particles when needed.
  2. The use of perfect forwarding (Fn&&) allows for efficient passing of the counter function.
  3. The conditional reservation prevents unnecessary allocations for coarse boundary particles.

To enhance clarity, consider adding a brief comment explaining the method's purpose:

/**
 * @brief Reserves memory for particles in the given array.
 * @param particles The particle array to reserve memory for.
 * @param counter A function that returns the number of particles to reserve.
 * @note Only reserves for regular refinement of domain particles.
 */
template<typename Fn>
static void _reserve(ParticleArray& particles, Fn&& counter)

This documentation would provide context for future maintainers and users of the class.


183-196: LGTM! Consider extracting count_expected to a named function for better readability.

The addition of count_expected lambda and its use with _reserve is an excellent optimization:

  1. It efficiently estimates the number of incoming particles.
  2. Using this estimate with _reserve can significantly improve performance by reducing reallocations.

To enhance readability and potentially reuse this logic, consider extracting the lambda to a named function:

std::size_t estimateIncomingParticles(const std::array<const ParticleArray*, 2>& particlesArrays,
                                      const SAMRAI::pdat::CellOverlap::BoxContainer& destBoxes) const
{
    std::size_t incoming_estimate = 0;
    for (auto const& destinationBox : destBoxes)
    {
        auto const coarseSplitBox = coarsen(phare_box_from<dim>(getSplitBox(destinationBox)));
        for (auto const& sourceParticlesArray : particlesArrays)
            incoming_estimate += sourceParticlesArray->nbr_particles_in(coarseSplitBox);
    }
    return incoming_estimate;
}

// Usage
_reserve(destDomainParticles, [&]() { return estimateIncomingParticles(particlesArrays, destBoxes); });

This change would improve code organization and make the estimation logic more reusable and testable.


Line range hint 204-265: LGTM! Consider further refactoring for improved readability and maintainability.

The updates to the particle refinement logic are excellent improvements:

  1. The use of std::copy_if enhances both performance and readability.
  2. The isInDest lambda effectively encapsulates the condition for particle inclusion.
  3. The logic for different split types is now more clearly separated.

To further improve the code, consider the following suggestions:

  1. Extract the particle copying logic into a separate method to reduce the nesting level:
template<typename DestArray>
void copyRefinedParticles(const std::array<Particle_t, nbRefinedPart>& refinedParticles,
                          DestArray& destArray, const auto& isInDest)
{
    std::copy_if(std::begin(refinedParticles), std::end(refinedParticles),
                 std::back_inserter(destArray), isInDest);
}

// Usage
if constexpr (putParticlesInCoarseBoundary)
{
    if constexpr (splitType == ParticlesDataSplitType::coarseBoundary)
        copyRefinedParticles(refinedParticles, destCoarseBoundaryParticles, isInDest);
    else if constexpr (splitType == ParticlesDataSplitType::coarseBoundaryOld)
        copyRefinedParticles(refinedParticles, destCoarseBoundaryOldParticles, isInDest);
    else
        copyRefinedParticles(refinedParticles, destCoarseBoundaryNewParticles, isInDest);
}
else
{
    copyRefinedParticles(refinedParticles, destDomainParticles, isInDest);
}
  1. Consider using a switch statement for the split type logic if there are more than these three cases.

  2. Remove or replace the commented-out std::cout statements with proper logging if debugging information is needed.

These changes would further enhance the code's readability and maintainability.

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 2abb9aa and 8c4fa42.

📒 Files selected for processing (4)
  • src/amr/data/particles/refine/particles_data_split.hpp (6 hunks)
  • tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (3 hunks)
  • tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (4 hunks)
  • tests/amr/data/particles/refine/test_split.cpp (3 hunks)
🚧 Files skipped from review as they are similar to previous changes (2)
  • tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp
  • tests/amr/data/particles/refine/test_split.cpp
🧰 Additional context used
📓 Path-based instructions (2)
src/amr/data/particles/refine/particles_data_split.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

🔇 Additional comments (4)
tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (3)

51-54: Excellent addition of type aliases for improved readability

The introduction of Splitter_t and ParticlesRefineOperator_t type aliases is a great improvement. These aliases encapsulate complex type information, making the code more readable and easier to maintain. The meaningful names clearly indicate their purpose, which will help developers understand the code more quickly.


104-104: Good use of the new type alias in refineOperator_ initialization

The modification to use ParticlesRefineOperator_t in the initialization of refineOperator_ is a positive change. It simplifies the code, improves readability, and maintains consistency with the newly introduced type aliases. This change enhances code maintainability without altering the underlying functionality.


81-81: Good addition of default ratio, but verify existing usage

Adding a default value of 2 for the _ratio parameter increases the flexibility of the BasicHierarchy constructor. This change allows the constructor to be called without explicitly providing a ratio, which can be convenient in some scenarios.

However, it's important to ensure that this change doesn't lead to unexpected behavior in existing code that may rely on explicitly setting the ratio.

Please verify that existing code using this constructor is not affected by this change. You may want to search for all instances where BasicHierarchy is instantiated to ensure they behave as expected with this new default value.

✅ Verification successful

Default _ratio parameter addition verified

The addition of a default value of 2 for the _ratio parameter in the BasicHierarchy constructor has been reviewed. The only instantiation found is within a test file, and using the default value does not introduce issues.

🏁 Scripts executed

The following scripts were executed for the analysis:

Script:

#!/bin/bash
# Search for BasicHierarchy instantiations
rg --type cpp "BasicHierarchy\s*<.*>\s*\([^)]*\)" -g '!test_particledata_refine_basic_hierarchy.hpp'

Length of output: 254

src/amr/data/particles/refine/particles_data_split.hpp (1)

27-31: LGTM! Good improvements to the enum class.

The changes to the ParticlesDataSplitType enum class are beneficial:

  1. Specifying std::uint8_t as the underlying type improves type safety and reduces memory usage.
  2. Explicitly setting coarseBoundary = 0 ensures a consistent starting value and makes the enum's behavior more predictable.

These modifications enhance the code's clarity and efficiency.

Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 0

🧹 Outside diff range and nitpick comments (2)
tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (2)

45-50: Improved function signature and const correctness.

The changes to the loadCell function are well-implemented:

  1. The new signature with void return type and ParticleArray<dimension>& particles parameter improves efficiency by modifying the array in-place.
  2. The use of const for _3diCell, middle, and delta enhances code clarity and prevents accidental modifications.

These changes align with good C++ practices and should improve performance slightly.

Consider using constexpr for middle and delta if their values are known at compile-time:

constexpr double middle = 0.5;
constexpr double delta  = 0.30;

202-214: New method domainParticlesForLevel added.

The new domainParticlesForLevel method is a useful addition for accessing domain particles at a specific level. It correctly iterates through patches and collects the necessary particle arrays.

Consider returning std::vector<std::reference_wrapper<ParticleArray<dimension>>> instead of raw pointers to improve safety and prevent potential issues with dangling pointers:

auto domainParticlesForLevel(std::shared_ptr<SAMRAI::hier::PatchHierarchy> const& hierarchy,
                             int levelNumber)
{
    std::vector<std::reference_wrapper<ParticleArray<dimension>>> particle_arrays;
    auto level = hierarchy->getPatchLevel(levelNumber);
    for (auto& patch : *level)
        for (auto const& [name, dataId] : dataToAllocate_)
            particle_arrays.emplace_back(
                std::dynamic_pointer_cast<ParticlesData<ParticleArray<dimension>>>(
                     patch->getPatchData(dataId))
                     ->domainParticles);
    return particle_arrays;
}

This change ensures that the caller can't accidentally delete the particle arrays and provides a safer interface.

📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL

📥 Commits

Files that changed from the base of the PR and between 8c4fa42 and 618e31d.

📒 Files selected for processing (1)
  • tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (3 hunks)
🧰 Additional context used
📓 Path-based instructions (1)
tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

📓 Learnings (1)
📓 Common learnings
Learnt from: PhilipDeegan
PR: PHAREHUB/PHARE#898
File: src/amr/data/particles/refine/particles_data_split.hpp:181-182
Timestamp: 2024-10-16T09:54:01.455Z
Learning: The codebase uses C++17 features such as class template argument deduction, allowing template arguments to be omitted in declarations like `std::array`.
🔇 Additional comments (1)
tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (1)

158-158: Simplified loadCell call in initializeLevelData.

The changes in the initializeLevelData method correctly implement the new loadCell function signature. The simplification of the loadCell call improves code readability while maintaining the existing logic for iterating over cells.

Comment on lines +50 to +51
EXPECT_EQ(domainParticles[0]->size(), 2800);
EXPECT_EQ(domainParticles[0]->capacity(), 4032);
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

diff still unexplained.

@PHAREHUB PHAREHUB deleted a comment from coderabbitai bot Dec 13, 2024
Copy link

@coderabbitai coderabbitai bot left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Actionable comments posted: 2

🧹 Nitpick comments (9)
src/amr/data/particles/refine/particles_data_split.hpp (1)

108-108: Correct spelling in comment.

There's a typo in the comment: 'Finaly' should be 'Finally', and 'cartesion' should be 'Cartesian'. Also, 'both patch' should be 'both patches'.

Apply this diff to correct the typos:

-    // Finaly we need the cartesion geometry of both patch.
+    // Finally, we need the Cartesian geometry of both patches.
src/amr/utilities/box/amr_box.hpp (4)

91-91: Consider using a wider integer type for the loop counter.

Using std::uint8_t as a loop counter may lead to unintended type promotions and limited range. It's generally safer to use std::size_t or unsigned int for loop indices.

Apply this diff to improve the loop counter type:

-    for (std::uint8_t di = 0; di < dim; ++di)
+    for (std::size_t di = 0; di < dim; ++di)

92-92: Improve readability by separating statements.

Using the comma operator to separate multiple statements on the same line can reduce readability. Consider splitting them into separate lines.

Apply this diff to enhance readability:

-        box.lower[di] *= refinementRatio, box.upper[di] *= refinementRatio;
+        box.lower[di] *= refinementRatio;
+        box.upper[di] *= refinementRatio;

98-98: Consider using a wider integer type for the loop counter.

Using std::uint8_t as a loop counter may lead to unintended type promotions and limited range. It's generally safer to use std::size_t or unsigned int for loop indices.

Apply this diff to improve the loop counter type:

-    for (std::uint8_t di = 0; di < dim; ++di)
+    for (std::size_t di = 0; di < dim; ++di)

99-99: Improve readability by separating statements.

Using the comma operator to separate multiple statements on the same line can reduce readability. Consider splitting them into separate lines.

Apply this diff to enhance readability:

-        box.lower[di] /= refinementRatio, box.upper[di] /= refinementRatio;
+        box.lower[di] /= refinementRatio;
+        box.upper[di] /= refinementRatio;
src/core/data/particles/particle.hpp (1)

42-42: Clarify the purpose of sizes array.

The sizes array holds hardcoded size values {52, 64, 76}, which may become outdated if the Particle structure changes. Consider documenting the purpose of these values or using sizeof(Particle<dim>) to compute sizes dynamically.

tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (1)

103-119: Consider extracting capacity calculation into a helper function.

The test case contains complex calculations for particle capacity. Consider extracting these calculations into a helper function for better maintainability.

+private:
+    auto calculateExpectedCapacity() const {
+        auto outer_cell_count = std::pow(50, 2) - std::pow(48, 2);
+        return particles.size() + (outer_cell_count * nbrParticlesPerCell * .1);
+    }
src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

183-188: Optimize lambda parameter in std::accumulate.

The lambda parameter sum should be passed by value instead of by const reference since it's a fundamental type.

-                                           [&](auto const& sum, auto const& density_value) {
+                                           [&](auto sum, auto const& density_value) {
tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (1)

171-174: Document the return type for better clarity

While the method implementation is correct, using auto return type without documentation makes it harder to understand the expected return type. Consider adding a documentation comment specifying the return type.

+    /**
+     * @brief Get domain particles for a specific level
+     * @param levelNumber The level number to get particles for
+     * @return std::vector<ParticleArray<dimension>*> Vector of particle arrays for the level
+     */
     auto domainParticlesForLevel(int levelNumber) const
     {
         return tagStrategy_->domainParticlesForLevel(hierarchy_, levelNumber);
     }
📜 Review details

Configuration used: .coderabbit.yaml
Review profile: CHILL
Plan: Pro

📥 Commits

Reviewing files that changed from the base of the PR and between 618e31d and e67ca0a.

📒 Files selected for processing (11)
  • src/amr/data/field/refine/electric_field_refiner.hpp (2 hunks)
  • src/amr/data/particles/refine/particles_data_split.hpp (6 hunks)
  • src/amr/utilities/box/amr_box.hpp (2 hunks)
  • src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (4 hunks)
  • src/core/data/particles/particle.hpp (3 hunks)
  • src/core/utilities/box/box.hpp (2 hunks)
  • tests/amr/data/particles/refine/CMakeLists.txt (1 hunks)
  • tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp (3 hunks)
  • tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (4 hunks)
  • tests/amr/data/particles/refine/test_split.cpp (3 hunks)
  • tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (2 hunks)
✅ Files skipped from review due to trivial changes (1)
  • src/amr/data/field/refine/electric_field_refiner.hpp
🚧 Files skipped from review as they are similar to previous changes (2)
  • tests/amr/data/particles/refine/test_split.cpp
  • tests/amr/data/particles/refine/test_particle_data_refine_tag_strategy.hpp
🧰 Additional context used
📓 Path-based instructions (6)
src/core/utilities/box/box.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/core/data/particles/particle.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/amr/utilities/box/amr_box.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

src/amr/data/particles/refine/particles_data_split.hpp (1)

Pattern **/*.hpp: Review the C++ code, point out issues relative to principles of clean code, expressiveness, and performance.

📓 Learnings (1)
src/amr/data/particles/refine/particles_data_split.hpp (1)
Learnt from: PhilipDeegan
PR: PHAREHUB/PHARE#898
File: src/amr/data/particles/refine/particles_data_split.hpp:181-182
Timestamp: 2024-11-12T08:25:26.527Z
Learning: The codebase uses C++17 features such as class template argument deduction, allowing template arguments to be omitted in declarations like `std::array`.
⏰ Context from checks skipped due to timeout of 90000ms (6)
  • GitHub Check: build (macos-latest)
  • GitHub Check: Analyze (cpp)
  • GitHub Check: build (macos-14)
  • GitHub Check: build (macos-13)
  • GitHub Check: Analyze (python)
  • GitHub Check: build (ubuntu-latest)
🔇 Additional comments (8)
src/amr/utilities/box/amr_box.hpp (1)

96-102: Ensure integer division in coarsen function is correct.

In the coarsen function, integer division is used to divide box.lower[di] and box.upper[di] by refinementRatio. If box.lower[di] or box.upper[di] are not multiples of refinementRatio, this can lead to unexpected results due to truncation. Please verify that these values are always divisible by refinementRatio or consider handling non-divisible cases to prevent potential errors.

src/core/data/particles/particle.hpp (1)

58-62: Avoid manual size comments that may become outdated.

While the size comments are informative, they might become inaccurate if the data types change. Consider using static assertions or sizeof to programmatically ensure and document member sizes.

tests/core/data/maxwellian_particle_initializer/test_maxwellian_particle_initializer.cpp (2)

15-18: LGTM! Clean namespace organization.

The namespace structure is well-organized, separating 1D and 2D test functions.


79-101: LGTM! Well-structured test class.

The 2D test class follows good testing practices with:

  • Clear type aliases
  • Proper initialization
  • Consistent member organization
src/core/data/ions/particle_initializers/maxwellian_particle_initializer.hpp (2)

Line range hint 47-57: LGTM! Good parameter organization.

The constructor parameters are well-organized with appropriate default values for densityCutOff and over_alloc_factor.


190-192: LGTM! Improved memory allocation strategy.

The capacity calculation combines both the expected size and an estimate for incoming particles from surface cells, which should help reduce reallocations.

tests/amr/data/particles/refine/CMakeLists.txt (1)

32-32: LGTM! Appropriate test configuration.

Using add_no_mpi_phare_test is correct for tests that don't require MPI functionality.

tests/amr/data/particles/refine/test_particledata_refine_basic_hierarchy.hpp (1)

51-54: Well-structured type aliases that improve readability!

The new type aliases effectively reduce template verbosity while following proper naming conventions.

Comment on lines +135 to +142
NO_DISCARD auto surface_cell_count() const
{
// assumes box never smaller than 3 in any direction
auto const shape_ = shape();
auto const nested = shape_ - 2;
return core::product(shape_) - core::product(nested);
}

Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

💡 Codebase verification

Add runtime validation for box size assumption.

The review comment is valid. The codebase shows no existing validation for the minimum box size requirement, and tests reveal that boxes smaller than 3 units are possible. The suggested runtime check is necessary to prevent undefined behavior in surface_cell_count().

🔗 Analysis chain

Add runtime validation for box size assumption.

The function assumes the box is never smaller than 3 in any direction, but this assumption is not enforced. Consider adding a runtime check.

 NO_DISCARD auto surface_cell_count() const
 {
     // assumes box never smaller than 3 in any direction
+    for (std::size_t i = 0; i < dim; ++i) {
+        if (shape()[i] < 3) {
+            throw std::invalid_argument("Box must be at least 3 units in each dimension");
+        }
+    }
     auto const shape_ = shape();
     auto const nested = shape_ - 2;
     return core::product(shape_) - core::product(nested);
 }
🏁 Scripts executed

The following scripts were executed for the analysis:

Error: No shell scripts found


Script:

#!/bin/bash
# Find box-related files
fd box

# Look for box class definition and size-related code
ast-grep --pattern 'class box'

# Search for any existing size validations in box-related files
rg "size|shape|dimension" -A 3 -B 3 src/core/utilities/box/

Length of output: 8065


Script:

#!/bin/bash
# Look for test cases involving box size or surface cells
rg -A 5 -B 5 "surface_cell_count|Box.*[<{].*[>}]" tests/core/utilities/box/test_box.cpp

# Check if there are any small box tests
rg "Box.*[<{].*[>}].*\{.*[123].*\}" tests/core/utilities/box/test_box.cpp

Length of output: 4878

@@ -74,7 +78,7 @@ class BasicHierarchy
}


explicit BasicHierarchy(int _ratio)
explicit BasicHierarchy(int _ratio = 2)
Copy link

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion

Rename parameter to follow C++ naming conventions

The parameter name _ratio starts with an underscore, which is reserved for implementation use in C++. Consider renaming it to ratio or gridRatio.

-    explicit BasicHierarchy(int _ratio = 2)
+    explicit BasicHierarchy(int ratio = 2)
📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
explicit BasicHierarchy(int _ratio = 2)
explicit BasicHierarchy(int ratio = 2)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants