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

Audio examples #644

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

Conversation

rothmichaels
Copy link
Contributor

No description provided.

example/lfo.cpp Outdated Show resolved Hide resolved
example/lfo.cpp Outdated Show resolved Hide resolved
example/lfo.cpp Outdated Show resolved Hide resolved
example/lfo.cpp Outdated Show resolved Hide resolved
example/lfo.cpp Outdated

musical_context m_context;
quantity<si::hertz, float> m_frequency;
quantity_point<angular::radian, float> m_phase{0.f};
Copy link
Owner

Choose a reason for hiding this comment

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

SI also has radian but does not have revolutions. I am not sure which one is the better use case considering that you use SI already.

Copy link
Contributor Author

@rothmichaels rothmichaels Dec 20, 2024

Choose a reason for hiding this comment

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

I'm not sure I have a strong opinion. I went with this both because of angular::revolutions and having a base dimension angle seems more natural than angle being m / m.

If you think it would be better for the example to use si::radian and 2π * si::radian instead of angle::revolutions. Any strong opinions?

I can also poll my colleagues (who are not on holiday) tomorrow about this to get their thoughts.

Copy link
Owner

Choose a reason for hiding this comment

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

This was just an FYI so you can use the best tool for the job. You know better what is needed than me 😉

Split out audio units and musical context functions into individual
headers to be used in future examples in addition to the LFO example.
Add an example of wrapping an unsafe OS or host application API with a
type-safe quanitites API.  The `audio::get_musical_context` API used in
this example will also be used in other audio examples.
@rothmichaels rothmichaels force-pushed the feature/audio-examples branch from b0fbe94 to aec61bc Compare December 20, 2024 00:28
example/audio/third_party_audio_api.h Outdated Show resolved Hide resolved
example/audio/wrapped_third_party_audio_api.h Outdated Show resolved Hide resolved

void set_period(QuantityOf<isq::time> auto period)
{
m_frequency = 1.f / value_cast<float>(period);
Copy link
Owner

Choose a reason for hiding this comment

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

You can also use inverse() from math.h that provides additional safety (https://aurora-opensource.github.io/au/main/reference/math/#inverse_as-inverse_in). However, it is not that mandatory as you are using floating-point representations.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Oh nice, I like this.

void set_period(QuantityOf<audio::beat_count> auto period)
{
std::cout << MP_UNITS_STD_FMT::format("Setting period to {} -- ", period);
set_period(value_cast<float>(period) / m_context.current_tempo);
Copy link
Owner

Choose a reason for hiding this comment

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

This value_cast might not be needed here as current_tempo is already a floating point.

void reset() { m_phase = phase_t{0.f * angular::radian}; }

private:
using phase_t = quantity_point<angular::radian, mp_units::default_point_origin(angular::radian), float>;
Copy link
Owner

Choose a reason for hiding this comment

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

Yes, it is unfortunate that a point origin must be provided when we want to specify a specific representation type. I have no ideas on how to improve here. I believe that for points, most people will use double, and the main interest will be to work against different origins. This is why I provided arguments in such order. But maybe I am wrong, or is there another way to improve here?

@JohelEGP and @chiphogg, do you have any ideas here?

Copy link
Collaborator

Choose a reason for hiding this comment

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

If the origin is an expression, {}.

Copy link
Owner

Choose a reason for hiding this comment

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

I am not sure what you mean by that?

Copy link
Collaborator

Choose a reason for hiding this comment

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

I meant something like this, but it turns out that it doesn't work with auto NTTPs, whereas it does with function parameters (https://cpp1.godbolt.org/z/hPY9KeYfa):

#include <initializer_list>
#include <type_traits>

template<int Origin = 1, class Rep = double>
struct point;

static_assert(std::is_same_v<point<{}, int>, point<0, int>>);

// duration equal to 2 measures of 4/4 music (i.e. 2 whole notes at
// the current tempo):
const auto beats = 2 * audio::whole_note;
const auto buffer_duration = value_cast<float>(beats) / context.current_tempo;
Copy link
Owner

Choose a reason for hiding this comment

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

value_cast is probably not needed here?

// of audio samples. In this example we will create a buffer with
// duration equal to 2 measures of 4/4 music (i.e. 2 whole notes at
// the current tempo):
const auto beats = 2 * audio::whole_note;
Copy link
Owner

Choose a reason for hiding this comment

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

It is up to you, and this is purely a matter of personal preference. 😉 AAA (Almost Always Auto) is a nice style, and I like it in many cases (especially not to repeat myself).

However, with time, I have learned to appreciate CTAD as it makes the code easier to reason about. Using quantity and quantity_point CTAD placeholders instead of auto probably makes the code easier for a new user to understand or maintain.

Copy link
Owner

Choose a reason for hiding this comment

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

const Quantity auto beats is also an option, but it is more to type and does not have any benefits here over just const quantity beats.

std::cout << MP_UNITS_STD_FMT::format("Filling buffer with values from LFO @ {}", sin_gen.get_frequency());
std::generate(begin(buffer_1), end(buffer_1), sin_gen);

assert(buffer_1.size() > 0u);
Copy link
Owner

Choose a reason for hiding this comment

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

I am not sure what this assert does as you set the vector's size explicitly in line 132. Did you mean reserve() and then using back_inserter in the algorithm?

Comment on lines 139 to 141
for (std::size_t i = 1u; i < buffer_1.size(); ++i) {
std::cout << MP_UNITS_STD_FMT::format(", {}", buffer_1[i]);
}
Copy link
Owner

Choose a reason for hiding this comment

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

Again, it is up to you but I would use a range-for here 😉

auto buffer_2 = buffer_t(buffer_1.size());
std::generate(begin(buffer_2), end(buffer_2), sin_gen);

return buffer_1 == buffer_2;
Copy link
Owner

Choose a reason for hiding this comment

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

The error in the CI is strange, but #include <vector> is missing, so it might be the reason why the comparison operator is not found on one of the configurations.

rothmichaels and others added 9 commits December 20, 2024 10:30
- Try including <vector> to fix CI error
- Use `mp_units::inverse`
- Use range-based for-loop
- Use CTAG and `quantity` instead of `auto`
I was suprised by the need to use `.in(one)` here as I would have
assumed the result was already in unit `one`.
Comment on lines +127 to +128
const quantity buffer_size =
quantity_cast<audio::sample_count>((buffer_duration * context.current_sample_rate).in(one));
Copy link
Owner

Choose a reason for hiding this comment

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

Is a quantity_cast required here? I assume that the implicit conversion did not compile? Did you try an explicit conversion?

Suggested change
const quantity buffer_size =
quantity_cast<audio::sample_count>((buffer_duration * context.current_sample_rate).in(one));
const quantity buffer_size =
audio::sample_count((buffer_duration * context.current_sample_rate).in(one));

Copy link
Owner

Choose a reason for hiding this comment

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

Isn't audio::sample a good unit here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Yes, audio::sample is a good unit here. This is what I had before the changes to dimensionless quantities:

const quantity buffer_size = (buffer_duration * context.current_sample_rate).in(audio::sample);

Maybe I didn't know the best way to do this after the dimensionless changes.

Copy link
Owner

Choose a reason for hiding this comment

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

I've fixed the previous audio example in the paper (https://godbolt.org/z/z7sPqe66e) and improved inverse(). Maybe this will give you some ideas on how to refactor this.

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.

3 participants