/chaos

Twitch Controls Chaos (USB Controller Interception-Based)

Primary LanguageJupyter NotebookGNU General Public License v3.0GPL-3.0

Twitch Controls Chaos (Controller Edition)

Chaos

"I liked watching your stream, those mods you have are awesome!" - Shannon Woodward aka ShannonIsLive, Voice Actress for Dina in TLOU2

This project was inspired after watching Twitch user DarkViperAU stream "Twich Controls Chaos" using ChaosModV. Here are a few differences between that and this version:

  • This directly intercepts controller inputs instead of being integrated with game mods
  • This can run on any game (with supported controllers), including on non-modded console games
  • This must run on compatible hardware capable of acting as a USB host and device, like a Raspberry Pi 4
  • Due to the hardware constraints, this has been written in C++ and Python for a linux environment

Note: This code is in desperate need of refactoring. It is highly functional but also hacky. I'm focusing on function in the meantime and will work on cleaning things up over time.

Currently this is hard-coded for The Last Of Us Part 2. Controller-based Chaos works very well for this game due to the massive amount of accessibility options, gameplay modifiers, and render modes. Eventually I will make this more portable for other games, starting with Uncharted 4/The Lost Legacy.

What is Chaos?

Watch a minute long video explanation of how users can interact with a controller. Twitch Clip

Notable Users of Chaos

This is a collection of streamers that have used chaos along with some example chaos clips from their streams

inabox44

RachyMonster

Feel free to check out my YouTube channel for videos that showcase Chaos.

Required Hardware

Here is an Amazon shopping list with items that are used to install and run chaos.

Important: The only controller supported is the Dualshock 4 Generation 2 - Has the lightbar visible from the front at the top of the touchpad - Model: CUH-ZCT2U

Note: The first generation Dualshock 4 controller can only send control inputs over bluetooth, making it unusable for this project.

Installation

Currently this setup is only supported on a Raspberry Pi 4 with 32-bit Raspian Lite, though other setups may work. Also, currently directories are hardcoded to look for /home/pi/chaos, so make sure to check out this project into your default account home directory. This may change later depending on project interest.

Step-by-step Setup Guide

  1. Flash Raspbian OS Lite (32-Bit) to your SD card.
  • I like to use this software tool to flash SD cards: Raspberry Pi Imager
    • Connect your SD card to your computer using an SD card reader
    • Select the SD card in the Raspberry Pi Imager
    • Under "Choose OS" select Raspberry Pi OS (other) -> Raspberry Pi OS Lite (32-bit)
    • Click on the "Write" button. If writing fails, simply try it again.
  1. Install the SD card into your Pi. Then proceed to connect a monitor and keyboard. Now apply power over the Pi's USB-C connector.

Now is the time to set up your Raspberry Pi's network. I highly recommend using ethernet for Chaos, though low-interferenced WiFi should be fine too. WiFi Instructions here.

  1. When your Pi boots log in using the following credentials:
  • Username: pi
  • Password: raspberry

Note: The password field will look like nothing is getting typed but it will be reading the password as you type it.

  1. If you have not already installed git (like on fresh installations of Raspbian Lite), run:
sudo apt update && sudo apt install git -y
  1. Run the following commands:
cd
git clone https://github.com/blegas78/chaos.git
cd chaos
./install.sh

Running the above final install command will take ~5 minutes

  1. A reboot is required to enable USB communication to hosts:
sudo reboot

Chaos should now be installed! Now all that is needed is configuration and setup for OBS and your particular game.

First Time Setup

Now that everything is installed, power up your Raspberry Pi and follow these steps:

  1. Your console will need to be set up to prefer USB communication over bluetooth for your controllers. On a PS5 do this by navigating to Settings (gear icon top right) -> Accessories -> Controllers -> Communication Method -> Use USB Cable

  2. On the same network as the Raspberry Pi, navigate your stream PC's browser to raspberrypi.local by clicking here. If that doesn't work you may need to figure out your pi's IP address from your router and enter that into your browser instead. Alternatively, run the command ifconfig on your Pi to get your Pi's IP address.

  3. Click on the link labeled "Interface" and navigate to the "Bot Setup" tab. Here you can enter your bot account credentials and your channel name. You can also use your broadcaster account if you do not have a secondary custom bot account.

  4. The other links next to "Interface" are overlays that can be added as a browser source in streaming applications like OBS. Open these links to view what they are, or copy and paste into a browser source on OBS. I prefer to have the browser sources in OBS set to refresh when not displayed so that I can refresh them easily. More of the OBS setup instruction are available in the Stream Setup guide.

OBS, Streamlabs

I recommend using things like a Chat Box and Color Correction filters to make chat more readable and change the colors to match your style:

Stream Setup

Running

  1. For each start, You must connect your USB controller to the lower left USB port on your Raspberry Pi. This is the USB 2.0 port furthest from the ethernet port and closest to circuit board. Then use a USB-C to USB-A cable to connect the Raspberry Pi's power to your console.

  2. Turn on your console. On a Playstation, pressing the PS button will turn on the console, which will then power the Raspberry Pi. During the Pi's boot process, the controller's bluetooth connection will let you navigate to your game. At some point, you'll see a USB connectivity notification, meaning that chaos is active.

  3. If OBS was already up, refresh your browser sources. Overlays should be active.

  4. Test your connection to your chat. Pull up your chat and try to vote. If a vote shows up on your overlay all is connected and ready to go!

  5. Lastly check that the controller is connected properly. Start up and load into your game so that you are controlling your character. Press the "Share" button to check that the VoteTimer resumes. If the timer runs and you can control your character, then all is connected and ready to go!

Usage

Outside of pausing/resuming, there's not much that needs to be done on the streamer's side. With all systems up and running, your viewers should be able to start voting and cause you all sorts of hell.

Pausing

Chaos initializes in a paused state. This paused state prevents any modifier from running, passing through all controls as normal. Pausing chaos will let you navigate things like your console's system menu without wreaking true havoc.

  • To resume Chaos, press the Share button.
  • To pause Chaos, press the Option button.

Note: This means that you will actively need to resume Chaos whenever you enter your game's pause menu.

Game-Specific Usage

Currently TLOU2 is hardcoded as the only supported game. See TLOU2 specific instructions here:

TLOU2 Usage

Contributors

Chaos would not be nearly as colorful or effective without community contributions, so thanks to everyone that has made this project better!

General

  • Aeathone Twitch
    • Informs me of my terrible curly braces <3
    • Suggested great design improvements to handling things like menuing
  • EspritDeCorpse Github Twitch
    • Reviewer
  • PrincessDiodes Github Twitch
    • Server-side gamepad viewer implementation
  • RachyMonster Twitch
    • Figured out how to make readable and spoiler free chat in the Stream Setup guide
  • ners_14 Twitch
    • Informed the idea of proportional voting from GTAV chaos

Modifier Ideas

Implemented:

TBD:

Known Issues/TODO:

The following features/bug will be implemented/fixed based on project interest over time.

USB hot-plugging

If the controller becomes disconnected, the chaos service must be restarted or a reboot must be performed. This is certainly annoying and can be fixed in code with an overhaul to usb-sniffify to allow hot-plugging.

USB crashing or delaying inputs over time on PS5

This can usually only currently be corrected with a reboot of the Pi. dmesg will show many errors and they are fine, but eventually the kernel will prevent raw-gadget from running. This may be a limitation of the raw-gadget Linux kernel module.

Configuration

Right now many of the timings and values are hard-coded, these can be slowly implemented in the webUI over time.

Video feedback

Implementing video feedback in a similar form to video-based load removers can aid in providing game-state information so that Chaos can know when it can/cannot perform certain actions. This would be a major overhaul and add complication to the game setup, so this is certainly a long-term stretch goal.

Semantic modifiers

Right now the chaos structure can only operate on pipelined button commands or apply direct controls. It cannot block any command on the controller output, meaning that some modifiers named "No shooting" truly mean "Disable passing R2". If a button remapping occurs then R2 could be sent through stacked modifiers involving button remapping, meaning that "No shooting" could not be accurate. For voting purposes, it would be nice to have a true semantic mapping for all modifiers. This will require adding more functions to modifiers, to be performed specifically at the end of the current chaos pipeline.

Design

The lowest level aspect of controller chaos is based on forwarding USB protocols using the Linux raw-gadget kernel module. For every USB request, Chaos duplicates the request and passes it along. However, in the case of messages corresponding to controller buttons/joysticks the data is passed to other processes that can meddle with the data. This forwarding infrastructure is done by using usb-sniffify, a library that combines raw-gadget and libusb.

The core of this Chaos implementation uses something called the ChaosEngine, written in C++ for speed. Stream overlays, the IRC Twitch bot, and vote tracking are built using Python. The overlays are built using Flexx. The IRC client code is unnecessarily custom (a library should be used).

Chaos uses softmax to select modifiers for the voting pool. This is implemented to reduce the probability that modifiers appear based on the number of times it has won. This reduces the likelihood that modifiers repeat to create more diversity in each moment however it is still possible to see repeat modifiers. Some viewers like Chaos Coach Cholocco will inform users to only vote on buff/visual modifiers during slow segments so that painful modifiers are more likely to appear during combat sections.

The ChaosEngine listens to the Python voting system using ZMQ with zmqpp. When a winning modifier comes in, ChaosEngine adds the modifier to list of active modifiers. After a set amount of time, the ChaosEngine will remove the modifier.

Each modifier can perform a set of actions at each of these transitions. There can be unique actions performed as the modifier is added, while it's currently active, when it ends, and can also perform asynchronous actions that are controller event-based. This effectively follows UML-style state machine with entry/do/exit/event actions.

ChaosEngine is designed to allow for flexibility in modifiers using the following principles:

  • Sniffing - Can read input being sent along the way
  • Interception - Can block pipelined commands
  • Modification - Replace message data including the type, id, and value
  • Injection - Can generate and send arbitrary messages without any controller events
  • State Sampling - See what is being applied versus what is trying to be sent
  • Direct Control - Can send commands directly to the output

Some other included helper utilities for modifiers include sequencers. With the above list and utilities, Chaos is capable of behaviors including button macros and remmaping. Such a framework can potentially be used as a TAS device, for "Twitch Plays ..." streams, or for modifying/boosting controller performance (could be considered cheating in multiplayer games).

Example Modifiers

Example modifier for applying inverted mode (Event Modification):

class Inverted : public Modifier {
public:
	static void regist() { Modifier::factory["Inverted"] = [](){return new Inverted();}; };
	bool tweak( DeviceEvent* event ) {
		if ((event->id == AXIS_RY ) && event->type == TYPE_AXIS) {
			event->value = -event->value;
		}
		return true;
	}
};

Example modifier for applying a control sequence once (Sequencing and Interception):

class TossMolly : public Modifier {
public:
	static void regist() { Modifier::factory["Toss A Molly"] = [](){return new TossMolly();}; };
	bool busy;
	
	void begin() {
		busy = true;
		Sequence sequence;
		sequence.disablejoysticks();
		sequence.addAxisPress(AXIS_DY, 1);
		sequence.addAxisPress(AXIS_DY, 1);
		sequence.addTimeDelay(2500);
		sequence.addButtonHold(BUTTON_R2);
		sequence.addTimeDelay(500);
		sequence.addButtonRelease(BUTTON_R2);
		sequence.send(dualshock);
		busy = false;
	}
	
	bool tweak( DeviceEvent* event ) {
		return !busy;
	}
};

Example modifier for adding time-functions to a joystick based on a button press (Interception, Modification, Sampling, Injection):

class MegaScopeSway : public Modifier {
public:
	static void regist() { Modifier::factory["Mega Scope Sway"] = [](){ return new MegaScopeSway();}; };
	
	std::map<int,int> axisToValue;
	std::map<int,int> offsetValue;
	
	void begin() {
		axisToValue[AXIS_RX] = dualshock->getState(AXIS_RX, TYPE_AXIS);
		axisToValue[AXIS_RY] = dualshock->getState(AXIS_RY, TYPE_AXIS);
		offsetValue[AXIS_RX] = 0;
		offsetValue[AXIS_RY] = 0;
	}
	
	void update() {
		DeviceEvent event;	// Event to inject
		event.type = TYPE_AXIS;
		
		bool applySway = dualshock->getState( BUTTON_L2, TYPE_BUTTON) != 0;
		
		double t = timer.runningTime()*3.0;
		double i = 1.0;
		
		for (std::map<int, int>::iterator it = axisToValue.begin(); it != axisToValue.end(); it++) {
			event.id = it->first;
			if (applySway) {
				offsetValue[event.id] = sin(((t+1.6)*i )*4.0) * JOYSTICK_MAX ;
				event.value = joystickLimit( it->second + offsetValue[event.id] );
				chaosEngine->fakePipelinedEvent(&event, this);
				i += 1;
			} else {
				offsetValue[event.id]  = 0;
			}
		}
	}
	
	void finish() {
		DeviceEvent event = {0,(short)axisToValue[AXIS_RX],TYPE_AXIS,AXIS_RX};
		chaosEngine->fakePipelinedEvent( &event, this);	
		event = {0,(short)axisToValue[AXIS_RY],TYPE_AXIS,AXIS_RY};
		chaosEngine->fakePipelinedEvent( &event, this);
	}
	
	bool tweak( DeviceEvent* event ) {
		if( event->type == TYPE_AXIS ) {
			if ( axisToValue.count(event->id) > 0) {
				axisToValue[event->id] = event->value;
				event->value = joystickLimit( event->value + offsetValue[event->id] );
			}
		}
		return true;
	}
};