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

udev and SDL Sensor and Touchpad #16188

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

Conversation

oshaboy
Copy link

@oshaboy oshaboy commented Feb 2, 2024

Description

Implemented Sensor and Touchpad Support for SDL and udev.

Related Issues

#16162

Reviewers

@gouchi @warmenhoven

Known Issues with the pull request

  • No Wiimote support yet.
  • Everything is hardcoded and not adjustable by the user. Which causes problems if an unrelated device such as a mouse takes device 0. Also if you swap the gamepads
    • Only the axes are hardcoded now.
    • SDL Device selection doesn't work.
    • Axes are no longer hardcoded but they aren't saved for some reason
  • Requires UDEV_TOUCH_SUPPORT to be enabled.
  • Untested with multiple Dualshock 4s/Dualsenses

@oshaboy
Copy link
Author

oshaboy commented Feb 2, 2024

The checks failed due to the UDEV_TOUCH_SUPPORT not being defined.

I'm not really sure what to do should I add a UDEV_SENSOR_SUPPORT macro for the sensor code?

@gouchi
Copy link
Member

gouchi commented Feb 2, 2024

To make some test

git clone https://github.com/libretro/RetroArch.git
cd RetroArch
git fetch origin pull/16188/head:pr16188
git checkout pr16188
./configure && make -j$(nproc)

@LibretroAdmin
Copy link
Contributor

You need to fix the failing CI tasks

@oshaboy
Copy link
Author

oshaboy commented Feb 2, 2024

Will do

@LibretroAdmin
Copy link
Contributor

Linux x86 pipeline still failing.

@gouchi
Copy link
Member

gouchi commented Feb 3, 2024

Linux x86 pipeline still failing.

It seems that sensor API has been introduced in SDL version 2.0.9. And we are using SDL 2.0.4 for the CI as we are building under Xenial.

@oshaboy
Copy link
Author

oshaboy commented Feb 3, 2024

Even if the build succeeds I don't want the pull request merged yet.

It works adequately now for testing but to be usable I need to add a lot of options to adjust sensitivity and mapping. The problem is RN the retropad stack assumes there's only one set of sensors... safe assumption to make on mobile. Not so much with sensor enabled controllers.

An example is that on the dualsense is the way it's set up by default you have to point the shoulder buttons up. Most people play games with the face buttons pointing up so it makes playing extremely unintuitive.

I only made the pull request so people will be aware that I am working on it and maybe get some more testing on it. Please don't merge it until it's complete.

@warmenhoven
Copy link
Contributor

You can tell GitHub to mark the PR as a draft, which will prevent it from being merged until you tell it that it's ready for review.

@oshaboy oshaboy marked this pull request as draft February 3, 2024 15:36
Cleanup
Sensitivity settings and Sensor Retropad selection implemented including menu options
@gouchi
Copy link
Member

gouchi commented Feb 3, 2024

I confirm DS4 is working with udev driver.

  1. Enable Auxiliary Sensor Input in Settings Tab > Input.
    auxiliary-sensor-input

  2. Select correct Sensor in Settings Tab > input > RetroPad Binds > Port 1 Controls > Sensor Index

sensor-index

If needed, you can adjust Sensivity in Settings Tab > Input > Gyroscope Sensitivity

sensivity

@LibretroAdmin
Copy link
Contributor

CI Linux (i686) still fails. Is SDL_Sensor only available on specific SDL versions? Can we have a compile-time ifdefs for this?

@gouchi
Copy link
Member

gouchi commented Feb 4, 2024

It seems there is a macro SDL_VERSION_ATLEAST so we could use SDL_VERSION_ATLEAST(2,0,9).

@oshaboy
Copy link
Author

oshaboy commented Feb 4, 2024

I was going to say that maybe we should drop support for ancient versions of SDL2, but turns out SDL 2.0.9 released in 2018 so it isn't that old.

@gouchi
Copy link
Member

gouchi commented Feb 4, 2024

but turns out SDL 2.0.9 released in 2018 so it isn't that old.

I am confused the release date indicates May 24, 2022 ?

Source

@oshaboy
Copy link
Author

oshaboy commented Feb 5, 2024

Idk the tags tab says 2018. But when you click on it it says 2022.

Maybe they have been pushing commits to it until then.

https://github.com/libsdl-org/SDL/tags?after=release-2.0.18

Edit: I found https://www.phoronix.com/news/SDL-2.0.9-Released confirming it's from 2018.

@oshaboy oshaboy marked this pull request as ready for review January 19, 2025 23:56
@oshaboy
Copy link
Author

oshaboy commented Jan 19, 2025

Yalla it's done. Kirby Tilt and Tumble works flawlessly now.

@oshaboy oshaboy marked this pull request as draft January 20, 2025 00:13
@oshaboy
Copy link
Author

oshaboy commented Jan 20, 2025

Actually there's still a design problem

@oshaboy oshaboy marked this pull request as ready for review January 20, 2025 02:41
@oshaboy oshaboy requested a review from JesseTG March 4, 2025 02:40
Copy link
Contributor

@JesseTG JesseTG left a comment

Choose a reason for hiding this comment

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

So far so good. However, we still need to see handlers for enabling and disabling the accelerometer and gyroscope; cores that don't need them shouldn't incur overhead in checking them. Thank you for your hard work.

unsigned input_config_parse_sensor(
unsigned id,
char *s,
void *conf_data);
Copy link
Contributor

Choose a reason for hiding this comment

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

Is there a reason this parameter is a void* despite not being used for any kind of polymorphism?

Copy link
Author

Choose a reason for hiding this comment

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

It's necessary for the API.

Copy link
Contributor

Choose a reason for hiding this comment

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

Are you sure? I'm searching for uses of it but I don't see any that would need it.

Copy link
Author

Choose a reason for hiding this comment

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

IDK all other functions that take a config_file_t pass it as a void *. If you want to refactor that that's on you. I am just following Chesterton's Fence principle.

Copy link
Contributor

Choose a reason for hiding this comment

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

Fair enough.


unsigned input_config_parse_sensor(
unsigned id,
char *s,
Copy link
Contributor

Choose a reason for hiding this comment

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

Could you make this parameter const to clarify your intent?

Copy link
Author

Choose a reason for hiding this comment

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

It shouldn't be const it uses char * s as a buffer. I might rename it to buffer instead.

Copy link
Contributor

Choose a reason for hiding this comment

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

s is only used with functions that take it as const. Is it possible you made a subtle mistake or typo somewhere?

Copy link
Author

Choose a reason for hiding this comment

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

Oh you're right. But all other functions such as fill_pathname_join_delim take a non const s. So I can't do it without discarding the qualifier.

/* NOTE: must be in sync with enum udev_input_dev_type */
static const char *g_dev_type_str[] =
{
"ID_INPUT_KEY",
"ID_INPUT_MOUSE",
"ID_INPUT_TOUCHPAD",
"ID_INPUT_TOUCHSCREEN"
"ID_INPUT_TOUCHSCREEN",
"ID_INPUT_ACCELEROMETER"
Copy link
Contributor

Choose a reason for hiding this comment

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

Should there be ID_INPUT_GYROSCOPE as well? Or am I misunderstanding something?

Copy link
Author

Choose a reason for hiding this comment

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

"ID_INPUT_ACCELEROMETER" is the whole sensor device according to udev

Comment on lines +4099 to +4100
case RETRO_SENSOR_GYROSCOPE_ENABLE:
case RETRO_SENSOR_ACCELEROMETER_ENABLE:
Copy link
Contributor

Choose a reason for hiding this comment

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

Did you forget to implement these?

Copy link
Author

@oshaboy oshaboy Mar 15, 2025

Choose a reason for hiding this comment

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

There is no need to. Linux already initializes them before they become udev devices.

Copy link
Contributor

Choose a reason for hiding this comment

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

Isn't there overhead in polling sensors and running the callbacks? Maybe it's not big, but cores that don't use it shouldn't have to pay for it. I would prefer it if you only opened and polled the sensors that the core actually asks for.

Copy link
Author

Choose a reason for hiding this comment

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

I have never seen a core poll the sensors frivolously. So AFAIK no core "pays" for it if it isn't enabled.

Copy link
Contributor

Choose a reason for hiding this comment

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

But udev itself is still polling the sensors and running the callback, right?

Copy link
Author

Choose a reason for hiding this comment

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

There's already code that assumes everything will be initialized and enabled in one go in a big for loop. If I don't initialize and open the sensor retroarch won't know what sensors are even available or even if sensors are available.

Copy link
Author

Choose a reason for hiding this comment

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

Actually I might be able to rig something one moment

Copy link
Author

Choose a reason for hiding this comment

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

Nope, sorry. udev_input_add_device just does too much. And I really don't want to make this pull request have another big refactor that will take a month to write and 6 months to verify.

Copy link
Author

Choose a reason for hiding this comment

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

But udev itself is still polling the sensors and running the callback, right?

udev will keep polling the sensors regardless of if we want to or not. And the "callback" is actually a file read operation. udev_input_poll already just reads all the open file descriptors regardless of whether they are requested or not. If you want to optimize the udev input driver go ahead but I think this is beyond the scope of this pull request.

Copy link
Author

@oshaboy oshaboy Mar 19, 2025

Choose a reason for hiding this comment

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

Honestly I think either way the branching to check whether the sensor was enabled or not will be more expensive than just running it unconditionally.

Comment on lines +9766 to +9767
snprintf(name,NAME_MAX_LENGTH,
"tmp name sensor bind %d", j);
Copy link
Contributor

Choose a reason for hiding this comment

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

Was this part supposed to be temporary?

Copy link
Author

Choose a reason for hiding this comment

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

Honestly I am not sure what the "name" parameter even does.

Copy link
Contributor

@JesseTG JesseTG left a comment

Choose a reason for hiding this comment

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

Now that I've had time to test and review this on my Steam Deck, I've found a lot of areas that need improvement.

My feedback mostly concerns the sdl2 input driver. There are numerous oversights that need to be addressed, mostly involving error-handling and variable assignments. But once I made some quick and dirty changes to my local copy, the sdl2 input driver started working consistently! The finish line's starting to come into sight.

I don't know as much about how udev works so I haven't been able to provide as much feedback with the time I have available. I can say that the udev driver is still not working with my Switch Pro controller's sensors. However, I suspect that if you give error-handling and edge cases some more attention then you may be able to uncover some issues.

int numJoysticks=0,numTouchDevices=0,numSensors=0;
int i; int sensor_count=0;

SDL_InitSubSystem(SDL_INIT_GAMECONTROLLER);
Copy link
Contributor

Choose a reason for hiding this comment

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

A few things:

  • Don't forget to call SDL_QuitSubSystem(SDL_INIT_GAMECONTROLLER) when tearing down the driver. SDL_InitSubSystem increases the refcount of any initialized flag, so there has to be corresponding cleanup at the end.
  • Please remember to handle a failure here.
  • You may need to include the SDL_INIT_SENSOR flag as well.

Copy link
Author

Choose a reason for hiding this comment

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

SDL_INIT_SENSOR is in SDL3 only. SDL2 doesn't have it.

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.

7 participants