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

IOS/USB: Emulate Wii Speak using cubeb #12769

Draft
wants to merge 20 commits into
base: master
Choose a base branch
from

Conversation

sepalani
Copy link
Contributor

@sepalani sepalani commented May 9, 2024

This PR is based on #12567.

I don't have much knowledge regarding the technical details of the implementation. I mainly fixed the Windows support, the heisenbug due to uninitialised members and finalised the config integration.

Ready to be reviewed and tested.


This change is Reviewable

@sepalani sepalani force-pushed the wii-speak branch 7 times, most recently from cf3916c to 46d7831 Compare May 12, 2024 19:40
@sepalani
Copy link
Contributor Author

I fixed a few more things. The Wii Speak Channel appears to work properly now (its initial test won't work though, as we don't emulate the echo reduction). I also added the Android support.

To test this PR, you'll need to enable the emulated Wii Speak under Tools > Emulated USB devices > Wii Speak and to check the two checkboxes to enable the emulation and to "connect" it. On Android, it's located under Settings > Config > Wii > Emulated USB Devices and you'll also need to enable the microphone permission for the app.

The Wii Speak Channel can be tested using a regular one (but by redirecting Nintendo domains to nothing like localhost, resulting in a skippable 20100 error) or via a Riiconnect24 patched version (AFAICT, WiiLink24 patcher doesn't seem to offer it currently, Wiimmfi WAD Patcher does). A Wii friend is required in order to be able to record Wii messages like this: https://www.youtube.com/watch?v=4kpNID36pW0

This PR can be tested on Monster Hunter 3 as well. You will need to patch the game to go online using alternative servers. You will need a friend in-game also using a Wii Speak (Dolphin or a real Wii) to be able to speak in a city.

I haven't checked this PR with other games yet, such as Animal Crossing. I'll when I'll get the time to do so. Otherwise, this PR is still ready to be reviewed & tested.

@sepalani sepalani force-pushed the wii-speak branch 2 times, most recently from 97daea9 to 3693fe5 Compare May 13, 2024 23:16
bool m_device_attached = false;
bool init = false;
std::unique_ptr<Microphone> m_microphone{};
const DeviceDescriptor m_device_descriptor{0x12, 0x1, 0x200, 0, 0, 0, 0x10,

Choose a reason for hiding this comment

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

Suggested change
const DeviceDescriptor m_device_descriptor{0x12, 0x1, 0x200, 0, 0, 0, 0x10,
const DeviceDescriptor m_device_descriptor{
.bLength = 0x12,
.bDescriptorType = 0x01,
.bcdUSB = 0x0200,
.bDeviceClass = 0x00,
...

Copy link
Contributor Author

Choose a reason for hiding this comment

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

AFAICT, this syntax isn't used in the current codebase for similar device classes. I wouldn't mind changing it in a follow up PR.

Copy link
Member

@JosJuice JosJuice left a comment

Choose a reason for hiding this comment

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

How does the permission prompt work on Android? Does cubeb automagically make the permission prompt show up?

Source/Android/app/src/main/AndroidManifest.xml Outdated Show resolved Hide resolved
@@ -923,4 +923,6 @@ It can efficiently compress both junk data and encrypted Wii data.
<string name="incompatible_figure_selected">Incompatible Figure Selected</string>
<string name="select_compatible_figure">Please select a compatible figure file</string>

<string name="emulate_wii_speak">Wii Speak</string>
<string name="disconnect_wii_speak">Mute Wii Speak</string>
Copy link
Member

Choose a reason for hiding this comment

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

"Mute" doesn't match what the setting is called in the code or in DolphinQt. Is this something that was overlooked, or is it intentional?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

It is intended since it is presented differently on PC and Android. On PC it has its own dedicated window (similar to the emulated Infinity base):
image

Whereas on Android, it's only a toggle settings below the one enabling the Wii Speak. So to make it clearer for the end-user I decided to rename it (as the effect and purpose of disconnecting it is not to record audio). I feel that having a "Wii Speak" toggle, plus a "Disconnect Wii Speak" toggle right under it sounds strange from a UI point of view.

I'm open to better ways of presenting this option to the user.

Copy link
Member

Choose a reason for hiding this comment

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

What does the setting do from the perspective of the emulated software? Does it disconnect the Wii Speak USB device, or does it just make the audio be silence? If it's the latter, I think both the Android UI and the PC UI are okay, but the internal name for the setting would make more sense as "mute" than "connect". "Connect" makes sense in the PC UI only because it's in the context of whether you want to connect a host microphone.

@sepalani sepalani force-pushed the wii-speak branch 2 times, most recently from dc72880 to 35436ac Compare June 4, 2024 18:44
@sepalani
Copy link
Contributor Author

sepalani commented Jun 4, 2024

@JosJuice

What does the setting do from the perspective of the emulated software? Does it disconnect the Wii Speak USB device, or does it just make the audio be silence? If it's the latter, I think both the Android UI and the PC UI are okay, but the internal name for the setting would make more sense as "mute" than "connect". "Connect" makes sense in the PC UI only because it's in the context of whether you want to connect a host microphone.

Ideally, it should have been used to disconnect the device. However, I couldn't find a way to do so and it had a side effect so I chose not to process the microphone data instead.

// Without a proper way to reconnect the emulated Wii Speak,
// this error after being raised prevents some games to use the microphone later.
//
// if (!IsMicrophoneConnected())
// return IPC_ENOENT;

bool Microphone::HasData() const
{
return m_samples_avail > 0 && Config::Get(Config::MAIN_WII_SPEAK_CONNECTED);
}

Yes, I think we should. It's probably best to do it both when turning on the setting and when the C++ code tries to use it. The former because that makes the purpose of the permission the most clear to the user, and the latter in case the permission has been revoked by the user (or automatically by the OS because the app hasn't been used for a while).

Apparently, enabling the permission after cubeb was initialised doesn't seem to work. I added a warning message before requesting the permission to warn the user about this issue. This part (especially the android code) needs to be properly reviewed as it's been a while since I've programmed for Android and I'm unsure that's the proper way to do it.

@sepalani sepalani force-pushed the wii-speak branch 2 times, most recently from 5d1ea67 to 0835422 Compare June 5, 2024 19:14
@JMC47
Copy link
Contributor

JMC47 commented Jul 3, 2024

Is anyone else planning to review this? Is this considered finished at this point?

@sepalani sepalani force-pushed the wii-speak branch 2 times, most recently from b21121e to 9023aa3 Compare July 8, 2024 09:07
@sepalani
Copy link
Contributor Author

sepalani commented Jul 8, 2024

Is anyone else planning to review this? Is this considered finished at this point?

The PR is finished. Though, it might be worth testing to ensure there's no oversight. Especially, on the Android part where I'm not very confident as it's working on my end but I'm unsure that the way I'm getting the activity is correct (or should I check the other callbacks too).

@JMC47
Copy link
Contributor

JMC47 commented Aug 7, 2024

Does anyone else want to review this? I can give this a test on Android, but I won't be reviewing the code.

@MayImilae
Copy link
Contributor

Are you still intending to add a microphone selector? Having a window with a single toggle is quite odd, as in that case you could just put the toggle itself into the file menu.

@sepalani
Copy link
Contributor Author

Are you still intending to add a microphone selector? Having a window with a single toggle is quite odd, as in that case you could just put the toggle itself into the file menu.

Doesn't the selector appear when the emulated device is enabled? I'm just using a layout similar to the one of the Infinity Base, another emulated USB device. I'm open to suggestion regarding this window design. The only constraint that it has to take into account (currently) is that it cannot enable or disable the emulated USB device when the emulation has started (i.e. it has to be chosen before launching the game).

image
image
image

@MayImilae
Copy link
Contributor

When I tested this I only saw what was in the top window. What scenario does that occur?

@sepalani
Copy link
Contributor Author

When I tested this I only saw what was in the top window. What scenario does that occur?

Currently, the microphone configuration groupbox is displayed only when the Emulate Wii Speak checkbox is checked. So the groupbox should (dis)appear when the checkbox state is changed. If there is a bug and it doesn't behave properly, then I'll have to investigate to see what's causing this.

Copy link
Member

@JosJuice JosJuice left a comment

Choose a reason for hiding this comment

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

Sorry for the delay in following up. I haven't been able to review all the USB and audio code in detail, but I've caught a few things.

@sepalani sepalani force-pushed the wii-speak branch 3 times, most recently from b938bcf to 987896a Compare September 22, 2024 09:54
@JMC47
Copy link
Contributor

JMC47 commented Sep 28, 2024

Last call for reviews on this. It's been waiting very long time and if I test it and the use cases are working on Linux and Windows, I will be merging it unless someone posts here and says they plan on reviewing it again or tells me not to merge it.

@sepalani sepalani force-pushed the wii-speak branch 2 times, most recently from 018c914 to 0352330 Compare September 28, 2024 19:16
@sepalani sepalani marked this pull request as draft November 16, 2024 14:10
@sepalani
Copy link
Contributor Author

I rebased this PR and added some kind of loudness level calculation. I'm not an audio expert so it might be worth checking that the formulas I'm using are correct.

I also switched this PR back to draft as I'm doing some more testing regarding the loudness level calculation.

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

Successfully merging this pull request may close these issues.

7 participants