diff --git a/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.cpp b/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.cpp index 1431b8b5219ef4622d21024132397690a81d9526..7b6e5d76a8568ab5f02c31b0b31eb49bbb33cd1b 100644 --- a/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.cpp +++ b/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.cpp @@ -69,15 +69,23 @@ void GamepadUnit::onConnectComponent() ARMARX_INFO << deactivateSpam(100000, std::to_string(dataTimestamp->getTimestamp())) << "No new signal from gamepad for " << age.toMilliSecondsDouble() << " milliseconds. Not sending data. Timeout: " << getProperty<int>("PublishTimeout").getValue() << " ms"; } }, 30); + sendTask->start(); ARMARX_TRACE; openGamepadConnection(); } +void GamepadUnit::vibrate(const ::Ice::Current&) +{ + ARMARX_INFO << "vibration!"; + js.executeEffect(); +} + bool GamepadUnit::openGamepadConnection() { if (js.open(deviceName)) { + ARMARX_TRACE; ARMARX_INFO << "opened a gamepad named " << js.name << " with " << js.numberOfAxis << " axis and " << js.numberOfButtons << " buttons."; if (js.numberOfAxis == 8 && js.numberOfButtons == 11) @@ -185,4 +193,3 @@ armarx::PropertyDefinitionsPtr GamepadUnit::createPropertyDefinitions() return armarx::PropertyDefinitionsPtr(new GamepadUnitPropertyDefinitions( getConfigIdentifier())); } - diff --git a/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.h b/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.h index 109ce81a17a98c5154b396e09064d9943be6ff0a..52c4791c51c37da03814e2a7295706431db99b13 100644 --- a/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.h +++ b/source/RobotAPI/drivers/GamepadUnit/GamepadUnit.h @@ -69,7 +69,8 @@ namespace armarx * Detailed description of class GamepadUnit. */ class GamepadUnit : - virtual public armarx::Component + virtual public armarx::Component, + virtual public GamepadUnitInterface { public: /** @@ -108,6 +109,8 @@ namespace armarx bool openGamepadConnection(); + void vibrate(const ::Ice::Current& = ::Ice::emptyCurrent) override; + private: GamepadUnitListenerPrx topicPrx; RunningTask<GamepadUnit>::pointer_type readTask; @@ -121,4 +124,3 @@ namespace armarx TimestampVariantPtr dataTimestamp; }; } - diff --git a/source/RobotAPI/drivers/GamepadUnit/Joystick.h b/source/RobotAPI/drivers/GamepadUnit/Joystick.h index 1f39074b7a1449f9d58b643d42be68f2c123c445..760563f81fd8837028a427515c5ea40c758e75ed 100644 --- a/source/RobotAPI/drivers/GamepadUnit/Joystick.h +++ b/source/RobotAPI/drivers/GamepadUnit/Joystick.h @@ -22,12 +22,17 @@ #pragma once -#include<linux/joystick.h> -#include<sys/stat.h> -#include<fcntl.h> +#include <cstdint> + +#include <fcntl.h> +#include <sys/poll.h> +#include <sys/stat.h> +#include <unistd.h> #include <ArmarXCore/core/Component.h> +#include <linux/joystick.h> + namespace armarx { @@ -36,21 +41,31 @@ namespace armarx private: int fd = -1; + int fdEvent = -1; js_event event; public: - std::vector<int16_t> axis; std::vector<bool> buttonsPressed; int numberOfAxis; int numberOfButtons; std::string name; - bool open(std::string const& deviceName) + bool + open(std::string const& deviceName) { + fd = ::open(deviceName.c_str(), O_RDONLY); + fdEvent = ::open("/dev/input/event2", O_RDWR | O_CLOEXEC); + + ARMARX_CHECK(fdEvent != -1); if (fd != -1) { + // ARMARX_INFO << "before"; + // executeEffect(); + // ARMARX_INFO << "after"; + + ioctl(fd, JSIOCGAXES, &numberOfAxis); ioctl(fd, JSIOCGBUTTONS, &numberOfButtons); name.resize(255); @@ -59,15 +74,21 @@ namespace armarx name = name.c_str(); buttonsPressed.resize(numberOfButtons, false); } + + ARMARX_INFO << "execute effect"; + executeEffect(); + return fd != -1; } - bool opened() const + bool + opened() const { return fd != -1; } - bool pollEvent() + bool + pollEvent() { int bytes = read(fd, &event, sizeof(event)); @@ -89,10 +110,163 @@ namespace armarx return true; } - void close() + void + executeEffect(int gain = 100, const int nTimes = 1) + { + // see https://docs.kernel.org/input/ff.html + ARMARX_CHECK(fdEvent >= 0); + + + // https://xnux.eu/devices/feature/vibrator.html + + int ret; + // pollfd pfds[1]; + int effects; + + // fd = open_event_dev("vibrator", O_RDWR | O_CLOEXEC); + // syscall_error(fd < 0, "Can't open vibrator event device"); + + ret = ioctl(fdEvent, EVIOCGEFFECTS, &effects); + ARMARX_CHECK(ret >= 0); + // syscall_error(ret < 0, "EVIOCGEFFECTS failed"); + + // ARMARX_CHECK(effects & FF_RUMBLE); + + // Set the gain of the device + { + // int gain; between 0 and 100 + struct input_event ie; // structure used to communicate with the driver + + ie.type = EV_FF; + ie.code = FF_GAIN; + ie.value = 0xFFFFUL * gain / 100; + + if (write(fdEvent, &ie, sizeof(ie)) == -1) + { + perror("set gain"); + } + } + + + ff_effect e; + // e.type = FF_RUMBLE; + // e.id = -1; + // e.replay.length = 5000; + // e.replay.delay = 500; + // e.u.rumble.strong_magnitude = 1; + + e.type = FF_PERIODIC; + e.id = -1; + e.replay.length = 5000; + e.replay.delay = 500; + e.u.periodic.waveform = FF_SQUARE; + e.u.periodic.period = 1000; + e.u.periodic.magnitude = 0xFF; + e.u.periodic.offset = 0xFF; + + ret = ioctl(fdEvent, EVIOCSFF, &e); + ARMARX_CHECK(ret >= 0); + + // syscall_error(ret < 0, "EVIOCSFF failed"); + + ARMARX_INFO << VAROUT(e.id); + + input_event play; + play.type = EV_FF; + play.code = static_cast<std::uint16_t>(e.id); + play.value = 1; + + ret = write(fdEvent, &play, sizeof(play)); + ARMARX_CHECK(ret >= 0); + + ARMARX_INFO << "Executing effect"; + + + for (int i = 0; i < 5; i++) + { + + input_event statusIe; // structure used to communicate with the driver + statusIe.type = EV_FF_STATUS; + statusIe.code = e.id; + // statusIe.value = 0; + + ret = write(fdEvent, &statusIe, sizeof(statusIe)); + + // ARMARX_CHECK() + // ret should be FF_STATUS_PLAYING + ARMARX_INFO << VAROUT(ret); + + sleep(1); + } + + + // syscall_error(ret < 0, "write failed"); + + ARMARX_INFO << "Executing effect"; + // sleep(6); + + + input_event stop; + stop.type = EV_FF; + stop.code = e.id; + stop.value = 0; + const int stopStatus = write(fdEvent, static_cast<const void*>(&stop), sizeof(stop)); + + + ret = ioctl(fdEvent, EVIOCRMFF, e.id); + ARMARX_CHECK(ret >= 0); + + // syscall_error(ret < 0, "EVIOCRMFF failed"); + + // close(fdEvent); + + + /**/ + // Set the gain of the device + // { + // // int gain; between 0 and 100 + // struct input_event ie; // structure used to communicate with the driver + + // ie.type = EV_FF; + // ie.code = FF_GAIN; + // ie.value = 0xFFFFUL * gain / 100; + + // if (write(fd, &ie, sizeof(ie)) == -1) + // { + // perror("set gain"); + // } + // } + + + // struct input_event play; + // struct input_event stop; + + // // upload request to device + // ff_effect effect; + // const auto uploadStatus = ioctl(fd, EVIOCSFF, &effect); + + // // Play n times + // play.type = EV_FF; + // play.code = effect.id; + // play.value = nTimes; + + // const int playStatus = write(fd, static_cast<const void*>(&play), sizeof(play)); + + // // Stop an effect + // stop.type = EV_FF; + // stop.code = effect.id; + // stop.value = 0; + // const int stopStatus = write(fd, static_cast<const void*>(&stop), sizeof(stop)); + + // // remove effect + // ioctl(fd, EVIOCRMFF, effect.id); + } + + void + close() { ::close(fd); fd = -1; } }; -} +} // namespace armarx diff --git a/source/RobotAPI/interface/units/GamepadUnit.ice b/source/RobotAPI/interface/units/GamepadUnit.ice index 671e1de9b99591f0d4e23ab923feac8a5e1d535d..c63ddcfdad36942cfcca6c65155c8e96c6847f79 100644 --- a/source/RobotAPI/interface/units/GamepadUnit.ice +++ b/source/RobotAPI/interface/units/GamepadUnit.ice @@ -62,8 +62,9 @@ module armarx bool rightStickButton; }; - interface GamepadUnitInterface extends armarx::SensorActorUnitInterface + interface GamepadUnitInterface // extends armarx::SensorActorUnitInterface { + void vibrate(); }; interface GamepadUnitListener @@ -78,4 +79,3 @@ module armarx }; }; -