-
Notifications
You must be signed in to change notification settings - Fork 67
/
Copy pathffb.h
344 lines (283 loc) · 12 KB
/
ffb.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
/*
Force Feedback Joystick
USB HID descriptors for a force feedback joystick.
This code is for Microsoft Sidewinder Force Feedback Pro joystick.
with some room for additional extra controls.
Copyright 2012 Tero Loimuneva (tloimu [at] gmail [dot] com)
Copyright 2023 Ed Wilkinson
MIT License.
Permission to use, copy, modify, distribute, and sell this
software and its documentation for any purpose is hereby granted
without fee, provided that the above copyright notice appear in
all copies and that both that the copyright notice and this
permission notice and warranty disclaimer appear in supporting
documentation, and that the name of the author not be used in
advertising or publicity pertaining to distribution of the
software without specific, written prior permission.
The author disclaim all warranties with regard to this
software, including all implied warranties of merchantability
and fitness. In no event shall the author be liable for any
special, indirect or consequential damages or any damages
whatsoever resulting from loss of use, data or profits, whether
in an action of contract, negligence or other tortious action,
arising out of or in connection with the use or performance of
this software.
*/
#ifndef _FFB_
#define _FFB_
#include <avr/io.h>
/* Type Defines: */
/** Type define for the joystick HID report structure, for creating and sending HID reports to the host PC.
* This mirrors the layout described to the host in the HID report descriptor, in Descriptors.c.
*/
// Maximum number of parallel effects in memory
#define MAX_EFFECTS 19 //Actually Max Effect ID , but effects IDs start at 0x02 so 1 less than this
//FFP can support 10 waveforms + 2 of each conditional = 18 not including other unsupported effect types
//Wheel limits?
// ---- Input
typedef struct
{
uint8_t reportId; // =2
uint8_t status; // Bits: 0=Device Paused,1=Actuators Enabled,2=Safety Switch,3=Actuator Override Switch,4=Actuator Power
uint8_t effectBlockIndex; // Bit7=Effect Playing, Bit0..7=EffectId (1..40)
} USB_FFBReport_PIDStatus_Input_Data_t;
// ---- Output
typedef struct
{ // FFB: Set Effect Output Report
uint8_t reportId; // =1
uint8_t effectBlockIndex; // 1..40
uint8_t effectType; // 1..12 (effect usages: 26,27,30,31,32,33,34,40,41,42,43,28)
uint16_t duration; // 0..32767 ms
uint16_t triggerRepeatInterval; // 0..32767 ms
uint16_t samplePeriod; // 0..32767 ms
uint8_t gain; // 0..255 (physical 0..10000)
uint8_t triggerButton; // button ID (0..8)
uint8_t enableAxis; // bits: 0=X, 1=Y, 2=DirectionEnable
uint8_t directionX; // angle (0=0 .. 255=360deg)
uint8_t directionY; // angle (0=0 .. 255=360deg)
// uint16_t startDelay; // 0..32767 ms
} USB_FFBReport_SetEffect_Output_Data_t;
typedef struct
{ // FFB: Set Envelope Output Report
uint8_t reportId; // =2
uint8_t effectBlockIndex; // 1..40
uint8_t attackLevel;
uint8_t fadeLevel;
uint16_t attackTime; // ms
uint16_t fadeTime; // ms
} USB_FFBReport_SetEnvelope_Output_Data_t;
typedef struct
{ // FFB: Set Condition Output Report
uint8_t reportId; // =3
uint8_t effectBlockIndex; // 1..40
uint8_t parameterBlockOffset; // bits: 0..3=parameterBlockOffset, 4..5=instance1, 6..7=instance2
uint8_t cpOffset; // 0..255
int8_t positiveCoefficient; // -128..127
// int8_t negativeCoefficient; // -128..127
// uint8_t positiveSaturation; // -128..127
// uint8_t negativeSaturation; // -128..127
// uint8_t deadBand; // 0..255
} USB_FFBReport_SetCondition_Output_Data_t;
typedef struct
{ // FFB: Set Periodic Output Report
uint8_t reportId; // =4
uint8_t effectBlockIndex; // 1..40
uint8_t magnitude;
int8_t offset;
uint8_t phase; // 0..255 (=0..359, exp-2)
uint16_t period; // 0..32767 ms
} USB_FFBReport_SetPeriodic_Output_Data_t;
typedef struct
{ // FFB: Set ConstantForce Output Report
uint8_t reportId; // =5
uint8_t effectBlockIndex; // 1..40
int16_t magnitude; // -255..255
} USB_FFBReport_SetConstantForce_Output_Data_t;
typedef struct
{ // FFB: Set RampForce Output Report
uint8_t reportId; // =6
uint8_t effectBlockIndex; // 1..40
int8_t start;
int8_t end;
} USB_FFBReport_SetRampForce_Output_Data_t;
typedef struct
{ // FFB: Set CustomForceData Output Report
uint8_t reportId; // =7
uint8_t effectBlockIndex; // 1..40
uint8_t dataOffset;
int8_t data[12];
} USB_FFBReport_SetCustomForceData_Output_Data_t;
typedef struct
{ // FFB: Set DownloadForceSample Output Report
uint8_t reportId; // =8
int8_t x;
int8_t y;
} USB_FFBReport_SetDownloadForceSample_Output_Data_t;
typedef struct
{ // FFB: Set EffectOperation Output Report
uint8_t reportId; // =10
uint8_t effectBlockIndex; // 1..40
uint8_t operation; // 1=Start, 2=StartSolo, 3=Stop
uint8_t loopCount;
} USB_FFBReport_EffectOperation_Output_Data_t;
typedef struct
{ // FFB: Block Free Output Report
uint8_t reportId; // =11
uint8_t effectBlockIndex; // 1..40
} USB_FFBReport_BlockFree_Output_Data_t;
typedef struct
{ // FFB: Device Control Output Report
uint8_t reportId; // =12
uint8_t control; // 1=Enable Actuators, 2=Disable Actuators, 3=Stop All Effects, 4=Reset, 5=Pause, 6=Continue
} USB_FFBReport_DeviceControl_Output_Data_t;
typedef struct
{ // FFB: DeviceGain Output Report
uint8_t reportId; // =13
uint8_t gain;
} USB_FFBReport_DeviceGain_Output_Data_t;
typedef struct
{ // FFB: Set Custom Force Output Report
uint8_t reportId; // =14
uint8_t effectBlockIndex; // 1..40
uint8_t sampleCount;
uint16_t samplePeriod; // 0..32767 ms
} USB_FFBReport_SetCustomForce_Output_Data_t;
// ---- Features
typedef struct
{ // FFB: Create New Effect Feature Report
uint8_t reportId; // =1
uint8_t effectType; // Enum (1..12): ET 26,27,30,31,32,33,34,40,41,42,43,28
uint16_t byteCount; // 0..511
} USB_FFBReport_CreateNewEffect_Feature_Data_t;
typedef struct
{ // FFB: PID Block Load Feature Report
uint8_t reportId; // =2
uint8_t effectBlockIndex; // 1..40
uint8_t loadStatus; // 1=Success,2=Full,3=Error
uint16_t ramPoolAvailable; // =0 or 0xFFFF?
} USB_FFBReport_PIDBlockLoad_Feature_Data_t;
typedef struct
{ // FFB: PID Pool Feature Report
uint8_t reportId; // =3
uint16_t ramPoolSize; // ?
uint8_t maxSimultaneousEffects; // ?? 40?
uint8_t memoryManagement; // Bits: 0=DeviceManagedPool, 1=SharedParameterBlocks
} USB_FFBReport_PIDPool_Feature_Data_t;
// Lengths of each report type
extern const uint16_t OutReportSize[];
// Handles Force Feeback data manipulation from USB reports to joystick's MIDI channel
void FfbSetDriver(uint8_t id);
// Initializes and enables MIDI to joystick using USART1 TX
void FfbInitMidi(void);
// Send "enable FFB" to joystick
void FfbSendEnable(void);
// Send "disable FFB" to joystick
void FfbSendDisable(void);
// Handle incoming data from USB
void FfbOnUsbData(uint8_t *data, uint16_t len);
// Handle incoming feature requests
void FfbOnCreateNewEffect(USB_FFBReport_CreateNewEffect_Feature_Data_t* inData, USB_FFBReport_PIDBlockLoad_Feature_Data_t *outData);
void FfbOnPIDPool(USB_FFBReport_PIDPool_Feature_Data_t *data);
// Utility to wait any amount of milliseconds.
// Resets watchdog for each 1ms wait.
void WaitMs(int ms);
// delay_us has max limits and the wait time must be known at compile time.
// function for making 10us delays that don't have be known at compile time.
// max delay 2560us.
void _delay_us10(uint8_t delay);
// Send raw data to the
void FfbSendData(const uint8_t *data, uint16_t len);
void FfbSendPackets(const uint8_t *data, uint16_t len);
void FfbPulseX1( void );
// Debugging
// <index> should be pointer to an index variable whose value should be set to 0 to start iterating.
// Returns 0 when no more effects
uint8_t FfbDebugListEffects(uint8_t *index);
// Effect manipulations
typedef struct
{
uint8_t midi; // disables all MIDI-traffic
uint8_t springs;
uint8_t constants;
uint8_t triangles;
uint8_t sines;
uint8_t effectId[MAX_EFFECTS];
} TDisabledEffectTypes;
extern volatile TDisabledEffectTypes gDisabledEffects;
uint8_t GetMidiEffectType(uint8_t id);
void FfbSendSysEx(const uint8_t* midi_data, uint8_t len);
uint8_t FfbSetParamMidi_14bit(uint8_t effectState, volatile uint16_t *midi_data_param, uint8_t effectId, uint8_t address, uint16_t value);
uint8_t FfbSetParamMidi_7bit(uint8_t effectState, volatile uint8_t *midi_data_param, uint8_t effectId, uint8_t address, uint8_t value);
uint16_t UsbUint16ToMidiUint14_Time(uint16_t inUsbValue);
uint16_t UsbUint16ToMidiUint14(uint16_t inUsbValue);
int16_t UsbInt8ToMidiInt14(int8_t inUsbValue);
uint16_t UsbPeriodToFrequencyHz(uint16_t period);
int8_t CalcGainCoeff(int8_t usbValue, uint8_t gain);
void FfbEnableSprings(uint8_t inEnable);
void FfbEnableConstants(uint8_t inEnable);
void FfbEnableTriangles(uint8_t inEnable);
void FfbEnableSines(uint8_t inEnable);
void FfbEnableEffectId(uint8_t inId, uint8_t inEnable);
// Bit-masks for effect states
#define MEffectState_Free 0x00
#define MEffectState_Allocated 0x01
#define MEffectState_Playing 0x02
#define MEffectState_SentToJoystick 0x04
#define USB_DURATION_INFINITE 0xFFFF
#define MIDI_DURATION_INFINITE 0x0000
#define USB_SAMPLEPERIOD_DEFAULT 0x0000
#define USB_TRIGGERBUTTON_NULL 0xFF
#define USB_EFFECT_CONSTANT 0x01
#define USB_EFFECT_RAMP 0x02
#define USB_EFFECT_SQUARE 0x03
#define USB_EFFECT_SINE 0x04
#define USB_EFFECT_TRIANGLE 0x05
#define USB_EFFECT_SAWTOOTHDOWN 0x06
#define USB_EFFECT_SAWTOOTHUP 0x07
#define USB_EFFECT_SPRING 0x08
#define USB_EFFECT_DAMPER 0x09
#define USB_EFFECT_INERTIA 0x0A
#define USB_EFFECT_FRICTION 0x0B
#define USB_EFFECT_CUSTOM 0x0C
#define USB_DCTRL_ACTUATORS_ENABLE 0x01
#define USB_DCTRL_ACTUATORS_DISABLE 0x02
#define USB_DCTRL_STOPALL 0x03
#define USB_DCTRL_RESET 0x04
#define USB_DCTRL_PAUSE 0x05
#define USB_DCTRL_CONTINUE 0x06
#define MAX_MIDI_MSG_LEN 27 /* enough to hold longest midi message data part, FFP_MIDI_Effect_Basic */
#define MAX_SHARE_DATA 12 /* enough bytes to hold all data that must be shared between Output reports for any effect type*/
/* start of midi data common for both pro and wheel protocols */
typedef struct {
uint8_t command; // 0x23 for pro, 0x20 for wheel
uint8_t waveForm; // different enumeration for pro/wheel
uint8_t unknown1; // always 0x7F
uint16_t duration; // unit=2ms
} midi_data_common_t;
typedef struct {
uint8_t state; // see constants <MEffectState_*>
uint8_t share_data[MAX_SHARE_DATA]; // All data to be shared between Output reports for calculating MIDI parameters coupled to multiple USB parameters
volatile uint8_t data[MAX_MIDI_MSG_LEN];
} TEffectState;
typedef struct
{
void (*EnableInterrupts)(void);
const uint8_t* (*GetSysExHeader)(uint8_t* hdr_len);
uint8_t (*DeviceControl)(uint8_t usb_control);
uint8_t (*UsbToMidiEffectType)(uint8_t usb_effect_type);
uint8_t (*EffectMemFull)(uint8_t new_midi_type);
void (*StartEffect)(uint8_t eid);
void (*StopEffect)(uint8_t eid);
void (*FreeEffect)(uint8_t eid);
void (*SendModify)(uint8_t effectId, uint8_t address, uint16_t value);
void (*ModifyDuration)(uint8_t effectState, uint16_t* midi_data_param, uint8_t effectId, uint16_t duration);
void (*ModifyDeviceGain)(uint8_t gain);
void (*CreateNewEffect)(USB_FFBReport_CreateNewEffect_Feature_Data_t* inData, volatile TEffectState* effect);
void (*SetEnvelope)(USB_FFBReport_SetEnvelope_Output_Data_t* data, volatile TEffectState* effect);
void (*SetCondition)(USB_FFBReport_SetCondition_Output_Data_t* data, volatile TEffectState* effect);
void (*SetPeriodic)(USB_FFBReport_SetPeriodic_Output_Data_t* data, volatile TEffectState* effect);
void (*SetConstantForce)(USB_FFBReport_SetConstantForce_Output_Data_t* data, volatile TEffectState* effect);
void (*SetRampForce)(USB_FFBReport_SetRampForce_Output_Data_t* data, volatile TEffectState* effect);
int (*SetEffect)(USB_FFBReport_SetEffect_Output_Data_t* data, volatile TEffectState* effect);
} FFB_Driver;
#endif // _FFB_