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

Lowpass Filter Improved Uniform Sampling Check #3978

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

Conversation

alexbeattie42
Copy link
Contributor

@alexbeattie42 alexbeattie42 commented Nov 27, 2024

Fixes issue #3968

Brief summary of changes

Implement an isUniform method to check if the vector is uniformly sampled. The implementation loosely follows the idea of the Matlab isUniform function. The implementation calculates the tolerance in this way:

isuniform verifies that the spacing between consecutive elements in numeric vector v does not deviate from the mean spacing by more than 4*eps(max(abs(v))), provided that the mean spacing is greater than that tolerance.

This may also fix this issue but I have not tested that myself yet.

Testing I've completed

Tested locally with this example and it fixes this issue. Will add unit tests soon.

Looking for feedback on...

@nickbianco and @aymanhab

  • do you have feedback on this method for checking the uniform sampling
  • where is the correct place for the uniform sampling check function to live. Maybe Signal?

CHANGELOG.md (choose one)

  • Updated

This change is Reviewable

Copy link
Member

@nickbianco nickbianco left a comment

Choose a reason for hiding this comment

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

Nice work @alexbeattie42!

My main suggestion is to expose isUniform by adding it to CommonUtilities, with some documentation about how the uniformity checks work. You could then reference that in the updated documentation for TableUtilities::filterLowpass.

Reviewed 1 of 1 files at r1, all commit messages.
Reviewable status: all files reviewed, 2 unresolved discussions (waiting on @alexbeattie42)


OpenSim/Common/TableUtilities.cpp line 126 at r1 (raw file):

template <typename T>
std::pair<bool,T> isUniform(const std::vector<T>& x) {

I think this would be a nice addition to CommonUtilities, as it might be useful for other table manipulation tools in OpenSim.


OpenSim/Common/TableUtilities.cpp line 189 at r1 (raw file):

    // Resample if the sampling interval is not uniform.
    if (!uniformlySampled) {

Could you add some documentation in the header file describing the conditions under which a table will be resampled?

@alexbeattie42 alexbeattie42 marked this pull request as ready for review December 5, 2024 08:00
@alexbeattie42 alexbeattie42 force-pushed the lowpass-resample branch 2 times, most recently from 85312a2 to b8a25c1 Compare December 5, 2024 09:56
@alexbeattie42 alexbeattie42 reopened this Jan 18, 2025
Copy link
Contributor Author

@alexbeattie42 alexbeattie42 left a comment

Choose a reason for hiding this comment

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

@nickbianco could you take another look?

Reviewable status: 0 of 6 files reviewed, 2 unresolved discussions (waiting on @nickbianco)


OpenSim/Common/TableUtilities.cpp line 126 at r1 (raw file):

Previously, nickbianco (Nick Bianco) wrote…

I think this would be a nice addition to CommonUtilities, as it might be useful for other table manipulation tools in OpenSim.

Done.


OpenSim/Common/TableUtilities.cpp line 189 at r1 (raw file):

Previously, nickbianco (Nick Bianco) wrote…

Could you add some documentation in the header file describing the conditions under which a table will be resampled?

Done.

Copy link
Member

@nickbianco nickbianco left a comment

Choose a reason for hiding this comment

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

Thanks @alexbeattie42! Made a handful of minor request, but overall looks pretty good.

Reviewed 6 of 6 files at r2, all commit messages.
Reviewable status: all files reviewed, 7 unresolved discussions (waiting on @alexbeattie42)


OpenSim/Common/CommonUtilities.h line 149 at r2 (raw file):

 * uniform, this value will be the mean step size. If the input is a one element
 * vector, the step size will be NaN since a valid step size cannot be
 * calculated with only 1 element.

I think you could reduce the indentation level for both of these list items.


OpenSim/Common/CommonUtilities.h line 153 at r2 (raw file):

 * @note The function uses a tolerance based on the maximum absolute value of
 * the first and last elements in the vector, scaled by machine epsilon. If the
 * vector is empty or contains only one element, the behavior is undefined.

Use left-aligned indent for the text in this note (like the note below.


OpenSim/Common/CommonUtilities.h line 156 at r2 (raw file):

 * @note The function implementation draws inspiration from Matlab's `isuniform`.
 *       See https://mathworks.com/help/matlab/ref/isuniform.html for more details.
 * details

OpenSim/Common/CommonUtilities.cpp line 77 at r2 (raw file):

    const double step_size = (end - start) / static_cast<double>((length - 1));
    for (int i = 0; i < length; ++i) {
        v[i] = std::fma(i, step_size,start);

nit:

Suggestion:

        v[i] = std::fma(i, step_size, start);

OpenSim/Common/TableUtilities.h line 97 at r2 (raw file):

     * rows is less than 4, or if the time intervals are not suitable for
     * resampling.
     *

nit:


OpenSim/Common/Test/testCommonUtilities.cpp line 135 at r2 (raw file):

        // Verify that the second value returned is the minimum step size found
        REQUIRE_THAT(result.second, Catch::Matchers::WithinAbs(0.03, tol));
    }

Not critical, but if you wanted to avoid "part 2" in the section headers here and below, you could scope the subtests like this (formatting isn't great here, but you get the idea):

SECTION("Non-uniform spacing") {
        {
            std::vector<double> vec3 = {1.0, 2.0, 3.5, 4.0};
            auto result = isUniform(vec3);
            REQUIRE(result.first == false);
            REQUIRE_THAT(result.second, Catch::Matchers::WithinAbs(0.5, tol));
      
            std::vector<double> vec4 = {1.0, 2.0, 3.0, 4.0, 5.1};
            result = isUniform(vec4);
            REQUIRE(result.first == false);
            REQUIRE_THAT(result.second, Catch::Matchers::WithinAbs(1.0, tol));
        }

        {
            std::vector<double> vec2 = {0.0, 0.5, 0.11, 0.16, 0.19, 0.24};
            auto result = isUniform(vec2);
      
            REQUIRE(result.first == false); // Should not be uniformly spaced
            // Verify that the second value returned is the minimum step size found
            REQUIRE_THAT(result.second, Catch::Matchers::WithinAbs(0.03, tol));
        }
    }

OpenSim/Common/Test/testCommonUtilities.cpp line 263 at r2 (raw file):

    REQUIRE_THAT(result_simtk.second, Catch::Matchers::WithinAbs(rate, tol));
    
}

nit: Add EOL line break.

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.

[Bug] Low Pass Filtering of Time IIR filter triggers a resample even if file is uniformly sampled
2 participants