Руль, джойстик и геймпад с обратной связью (Force Feedback)

Поводом для этой статьи явился, принесенный мне на ремонт, руль GW21-FB китайского производства. Обычно в устройствах такого происхождения ужасная механика, дешевые электролитические конденсаторы, не качественная пайка и ремонт не вызывает проблем. Иногда выходит из строя контроллер (как правило бескорпусный, залитый компаундом), в этом случае приходится менять всю плату, т.е. создавать ее заново, т.к. купить ее негде. Если руль (джойстик, геймпад) без обратной связи, то схема получается небольшая и что-либо изобретать нет необходимости — примеров реализации таких устройств в сети предостаточно. Например: Mjoy16 , Analog Joystick

В данном случае обратная связь должна быть, и я, походив по Интернету, с удивлением обнаружил, что ни одной реализации с открытым кодом нет! Многие самодельщики (в основном англо- и немецкоязычные) брались за это, но безрезультатно. Ну, что же, попробуем восполнить этот пробел. И попытаемся сделать это дешево, иначе ремонт может стать просто нерентабельным.

Далее последуют нудные разъяснения, кому это скучно, могут сразу перейти в конец статьи и скачать архив с готовыми схемами, платой, исходниками и прошивками. 

Итак, для начала, как же работает «обратная связь»или точнее FFB (Force feedback)? Есть два ее вида — Вибрационная и Активная. В первом случае в устройство просто встроен вибромотор, такой же как в сотовом телефоне, только немного помощнее. В определенные моменты игры, например при столкновении, наезде на препятствие, этот мотор включается и руль (джойстик, геймпад) начинает вибрировать.

vibro.jpg

На этом фото показан вибромотор в ручке джойстика. На ось мотора напрессован груз со смещенным центром тяжести. При вращении груза мотор вместе с ручкой начинает вибрировать. Достоверность тактильных ощущений при этом, естественно, не велика, поэтому чаще ставят два мотора. Они одинаковые, но грузы на них разные, поэтому у одного частота вибрации выше, но амплитуда ниже, у другого наоборот — частота ниже, а амплитуда выше. Кроме того, моторы расположены под углом друг к другу, поэтому и направление вибрации различно. Комбинируя одновременное и раздельное включение моторов можно несколько разнообразить тактильные ощущения.

rumblepad-3.jpg

Однако, частоту, амплитуду и, главное, направление вибраций, при таком подходе, варьировать крайне сложно, поэтому в устройствах более высокого класса применяется Активная система FFB. В рулях это мотор, соединенный с колесом руля зубчатой или ременной передачей:

actlabsaxe.jpg

mecaforce.jpg

В джойстиках применяют два мотора — по осям X и Y:

joy-1.jpg

Я не для красоты привожу эти картинки. Возможно кто-то захочет доработать свой «безответный» аппарат системой FFB и они могут послужить наглядным пособием для этого. Как видите, ничего заумно-хитрого здесь нет…

Итак, с механикой все понятно, управление моторами не представляет особой сложности, но нужно знать как и когда их включать/выключать. Требуется получить по шине USB информацию от компьютера. Обычный руль, джойстик, геймпад это HID-устройство, даже не требующее драйвера, они есть в системе, но для FFB все сложнее. Нужно либо представиться системе устройством, способным самостоятельно формировать сигнал управления мотором, получая от компьютера все данные (длина волны, амплитуда, фаза и т.д.), либо использовать драйвер-преобразователь.

В первом случае потребуется постоянно принимать большие объемы информации и иметь контроллер с функциями DSP. Кроме того, нужно глубоко изучить систему формирования сигнала для написания управляющей прошивки при том, что для выставления правильных коэффициентов для DSP потребуется оригинал для сравнения.

Во втором случае можно использовать драйвер готового устройства. Он сам выполнит все нужные преобразования, а нам останется только принимать мгновенные значения силы и направления вращения и подавать их на мотор.

Собственно изначально задача была недорого и без головной боли отремонтировать (переделать) сабж, поэтому выбираем второй вариант. И без стеснений берем драйвер от фирмы Logitech. К концу повествования мы создадим несколько рулей, но начнем с самой сложной модели — G25:

Logitech G25

Модель топовая, имеет три педали и 22 кнопки управления. Точнее говоря, из-за рычага переключения передач их кажется меньше, но информация о его положении передается как отдельные кнопки, а нам для выбора схемы важно знать их максимальное число, поэтому запомним — 22 кнопки + 3 педали.

Вот мы и подошли к схемной реализации. Сначала определимся с контроллером:

В современных рулях, джойстиках применяют высокоскоростной протокол USB 2.0, но мы попробуем обойтись низкоскоростным 1.1. Во первых принимаемый/передаваемый объем информации не столь уж велик, а во вторых микроконтроллеры с аппаратным USB довольно дороги, да и конвертеры USB/RS232 тоже. Попробуем применить «ширпотреб» — ATmega8 с программным вариантом реализации USB.

Прежде чем составить схему управления мотором FFB, требуется выяснить, в каком виде драйвер выдает эту информацию. Естественно, никакой документации на него нигде нет, обращение в службу поддержки завершилось отказом — протокол обмена секрет фирмы. Пришлось создать макет устройства, пытаться передавать драйверу данные и изучать его ответы. После нескольких дней мучений удалось выяснить, что комбинация

11 08 ХХ 80 00 00 00

означает включение мотора FFB, при этом ХХ определяет силу и направление вращения. ХХ принимает значение от 0 до 255, при этом 128 — «начало координат», все, что меньше — вращение влево, больше- вращение вправо. Т.е. данные можно сразу подавать на 8-битный ШИМ с мостовой схемой включения мотора. Комбинация

13 00 00 00 00 00 00

означает выключение мотора. Это тоже важно, т.к. при ШИМ=128 мотор, конечно, не крутится, но ток через него идет и, если уж компьютер разрешает, то лучше ток отключать.

Итак со схемой теперь все понятно:

Схема G25

Чтобы меньше сверлить отверстий в плате, применим SMD резисторы и конденсаторы:

SDC18527.jpg

Прошивку контроллера напишем в среде Bascom-AVR . Для программной реализации USB воспользуемся библиотекой SWUSB.LBX . Для удобства работы оформим дескрипторы и константы USB отдельными файлами, соответственно Descriptors.inc и USB_config.inc.

Сразу зададим начальное состояние портов ввода/вывода:

 Ddrb = &B00011000   'направление линий порта B: 1 - на вывод  Portb = &B00100111   'исходное состояние линий B  Ddrc = &B00110000   'направление линий порта C: 1 - на вывод  Portc = &B00000000   'исходное состояние линий C  Ddrd = &B00010011   'направление линий порта D: 1 - на вывод  Portd = &B11100000   'исходное состояние линий D

Назначим выводы D+ и D- для USB (D+ должна быть на INT0):

 _usb_port Alias Portd  _usb_pin Alias Pind  _usb_ddr Alias Ddrd  Const _usb_dplus = 2  Const _usb_dminus = 3

Зададим PID и VID, соответствующие Logitech G25:

 Const _usb_vid = &H046D   'Vendor ID  Const _usb_pid = &HC299   'Product ID

В нашем распоряжении всего 8 байт для передачи хосту, а устройство имеет , как мы помним, 22 кнопки, руль и три педали. Плюс к этому нужно еще передать байт «Combined Pedals» — как выяснилось, при активации/деактивации этой опции драйвер не передает никаких управляющих сигналов устройству и даже не пытается скомбинировать этот режим из имеющейся информации о педалях. Оригинального дескриптора достать не удалось, да он нам и не помог бы, ведь настоящий G25 работает в USB 2.0 и не ограничен пространством — может передавать до 64 байт за раз. Поэтому пришлось ограничить разрядность данных для педалей до 7 бит, а для руля до 12 бит, и репорт-дескриптор в финале получился таким:

    Data &H05 , &H01   ' Usage Page (Desktop),     Data &H09 , &H04   ' Usage (Joystik),     Data &HA1 , &H01   ' Collection (Application),     Data &HA1 , &H02   '   Collection (Logical),     Data &H95 , &H01   '   Report Count (1 field),     Data &H75 , &H0C   '   Report Size (12 bit),     Data &H14   '   Logical Minimum (0),     Data &H26 , &HFF , &H0F   '   Logical Maximum (4095),     Data &H34   '   Physical Minimum (0),     Data &H46 , &HFF , &H0F   '   Physical Maximum (4095),     Data &H09 , &H30   '   Usage (X),     Data &H81 , &H02   '   >>>> Input (Variable), *** Wheel ***     Data &H05 , &H01       '   Usage Page (Desktop),     Data &H95 , &H01   '   Report Count (1 field),     Data &H75 , &H04   '   Report Size (4 bit),     Data &H25 , &H07   '   Logical Maximum (7),     Data &H46 , &H3B , &H01   '   Physical Maximum (315),     Data &H65 , &H14   '   Unit (Degrees),     Data &H09 , &H39   '   Usage (Hat Switch),     Data &H81 , &H42   '   >>>> Input (Variable), *** Hat ***     Data &H95 , &H13   '   Report Count (19 fields),     Data &H75 , &H01   '   Report Size (1 bit),     Data &H25 , &H01   '   Logical Maximum (1),     Data &H45 , &H01   '   Physical Maximum (1),     Data &H05 , &H09   '   Usage Page (Button),     Data &H19 , &H01   '   Usage Minimum (01h),     Data &H29 , &H13       '   Usage Maximum (13h),     Data &H81 , &H02         '   >>>> Input (Variable), *** Buttons ***     Data &H05 , &H01       '   Usage Page (Desktop),     Data &H95 , &H01       '   Report Count (1 field),     Data &H75 , &H07      '   Report Size (7 bit),     Data &H14      '   Logical Minimum (0),     Data &H26 , &H7F , &H00         '   Logical Maximum (127),     Data &H34         '   Physical Minimum (0),     Data &H46 , &H7F , &H00         '   Physical Maximum (127),     Data &H09 , &H32      '   Usage (Z),     Data &H81 , &H02      '   >>>> Input (Variable), *** Accelerator ***     Data &H95 , &H01      '   Report Count (1 field),     Data &H75 , &H07      '   Report Size (7 bit),     Data &H14         '   Logical Minimum (0),     Data &H26 , &H7F , &H00         '   Logical Maximum (127),     Data &H34         '   Physical Minimum (0),     Data &H46 , &H7F , &H00       '   Physical Maximum (127),     Data &H09 , &H35         '   Usage (Rz),     Data &H81 , &H02         '   >>>> Input (Variable), *** Brake ***     Data &H95 , &H01      '   Report Count (1 field),     Data &H75 , &H07         '   Report Size (7 bit),     Data &H14         '   Logical Minimum (0),     Data &H26 , &H7F , &H00      '   Logical Maximum (127),     Data &H34         '   Physical Minimum (0),     Data &H46 , &H7F , &H00      '   Physical Maximum (127),     Data &H09 , &H36         '   Usage (Slider),     Data &H81 , &H02       '   >>>> Input (Variable), *** Clutch ***     Data &H95 , &H01         '   Report Count (1 field),     Data &H75 , &H08         '   Report Size (8 bit),     Data &H14         '   Logical Minimum (0),     Data &H26 , &HFF , &H00      '   Logical Maximum (255),     Data &H34         '   Physical Minimum (0),     Data &H46 , &HFF , &H00      '   Physical Maximum (255),     Data &H09 , &H31          '   Usage (Y),     Data &H81 , &H02         '   >>>> Input (Variable), *** Combined pedals ***     Data &HC0          '   End Collection,     Data &HA1 , &H02         '   Collection (Logical),     Data &H09 , &H02         '   Usage (02h),     Data &H95 , &H07         '   Report Count (7 fields),     Data &H91 , &H02      '   <<<< Output (Variable), *** FFB & Control ***     Data &HC0         '   End Collection,     Data &HC0   ' End Collection

Т.е. получилась дискретность педалей 0,78% , а руля 0,024% . Не так уж и плохо.

Для составления репорт-дескриптора можно воспользоваться справочником HID Usage Tables или специальной программой HID Descriptor tool.

Дескриптор продукта и изготовителя не важны, можно поставить любые — драйвер все равно вместо них подставит в систему название «Logitech G25».

В основной программе принимаем данные от хоста и по первому байту сортируем: если $11 (dec 17), то включаем мотор и выставляем ШИМ, если $13 (dec 19) — выключаем мотор:

    Select Case _usb_rx_buffer(2)   'Тип данных     Case 17     If _usb_rx_buffer(3) = 8 Then   'Данные ШИМ для мотора     Pwm2 = _usb_rx_buffer(4)     Ffb_motor = 1   'Включение мотора     End If     Case 19   'Выключение мотора     Pwm2 = 128     Ffb_motor = 0     End Select

Частоту ШИМ задаем 375кГц (12МГц/32), т.о. частота на выходе будет около1,5кГц (375/256):

 Config Timer2 = Pwm , Prescale = 32 , Compare Pwm = Clear Up  Tccr2.3 = 1     Tccr2.6 = 1

Выше частоту выставлять не рекомендую — транзисторы моста начнут греться, т.к. для их раскачки применен самый примитивный вариант (для удешевления конструкции). При этом в близи среднего положения руля мотор воспроизводит частоту 1,5кГц и для борьбы с этим эффектом последовательно с мотором включен дроссель, а параллельно конденсатор. Если имеются в наличии, то лучше для раскачки применить стандартные драйверы верхнего и нижнего плеч CMOS, тогда частоту ШИМ можно повысить до максимума (47кГц), не опасаясь перегрева и исключить дроссель.    

АЦП в ATmega8 10-разрядный, поэтому для получения 12-разрядного значения сложим 4 последовательных значения с потенциометра колеса руля, заодно таким образом уменьшив флуктуации сигнала:

  Wheel = 0   For I = 1 To 4        Temp_word = Getadc(0)     Wheel = Wheel + Temp_word   Next I

Состояние кнопок определим, выставляя поочередно на линиях Row0… Row3 низкий уровень. Резисторы, включенные в линии кнопок, защищают порты контроллера от короткого замыкания при одновременном нажатии двух и более кнопок. Программно для этого так же предусмотрены защитные временные интервалы.

Разъем GEAR дублирует кнопки S1 и S2 — его наличие необязательно.

На вторую ножку разъема кнопок выведен порт С3 — сюда можно будет подключить светодиод (пока не задействовано).

Данные переключателя видов «HAT» перед передачей хосту нужно перевести в градусы. Дискретность — 45°, при этом отжатое состояние соответствует 360°. Кроме того, эти данные нужно вписать в свободный верхний нибл старшего байта колеса руля, т.к. мы так задали в дескрипторе:

    Select Case Hat     Case 0 : Wheel.15 = 1   'Hat не нажат     'Case 1   'нажат / , все биты = 0     Case 2 : Wheel.13 = 1   'нажат >     Case 3 : Wheel.12 = 1   'нажаты / и >     Case 4 : Wheel.14 = 1   'нажат /     Case 6     Wheel.12 = 1 : Wheel.13 = 1   'нажаты > и /     Case 8     Wheel.13 = 1 : Wheel.14 = 1   'нажат <     Case 9     Wheel.12 = 1     Wheel.13 = 1 : Wheel.14 = 1   'нажаты < и /     Case 12     Wheel.12 = 1 : Wheel.14 = 1   'нажаты < и /     End Select

Потенциометры педалей опрашиваем аналогично колесу руля, так же 4 раза для снижения флуктуаций, затем делим на 32 (сдвигом на 5 разрядов вправо). В результате получаем данные в масштабе 7 бит. Далее создаем «Combined Pedals» — если не нажата педаль «тормоза», то она равна данным педали «газа». Если нажат «тормоз», то равна его инверсии — так требует драйвер.

   Accelerator = 0 : Brake = 0 : Clutch = 0     For I = 1 To 4        Temp_word = Getadc(1)   'педаль "Сцепление"     Clutch = Clutch + Temp_word     Temp_word = Getadc(2)   'педаль "Тормоз"     Brake = Brake + Temp_word     Temp_word = Getadc(3)   'педаль "Газ"     Accelerator = Accelerator + Temp_word     Next I     Shift Clutch , Right , 5   'масштабирование до 127     Shift Brake , Right , 5   'масштабирование до 127     Shift Accelerator , Right , 5   'масштабирование до 127       Combined_pedals = Accelerator     If Brake < 123 Then        Combined_pedals = 255 - Brake     End If

Далее следует пересортировать биты данных в соответствии с заданной очередностью в репорт-дескрипторе. Обычно такого делать не требуется, но в данном случае мы вынужденно пошли на такие сложности из-за стесненных условий. Зато мы уместили в пакете большой объем информации. После пересортировки передаем данные хосту:

    Buffer_6.3 = Accelerator.0   'пересортировка данных для отправки     Buffer_6.4 = Accelerator.1     Buffer_6.5 = Accelerator.2     Buffer_6.6 = Accelerator.3     Buffer_6.7 = Accelerator.4     Buffer_7.0 = Accelerator.5     Buffer_7.1 = Accelerator.6     Buffer_7.2 = Brake.0     Buffer_7.3 = Brake.1     Buffer_7.4 = Brake.2     Buffer_7.5 = Brake.3     Buffer_7.6 = Brake.4     Buffer_7.7 = Brake.5     Buffer_8.0 = Brake.6     Buffer_8.1 = Clutch.0     Buffer_8.2 = Clutch.1     Buffer_8.3 = Clutch.2     Buffer_8.4 = Clutch.3     Buffer_8.5 = Clutch.4     Buffer_8.6 = Clutch.5     Buffer_8.7 = Clutch.6        '------------------------- Отправка данных в ПК --------------------------     _usb_tx_buffer2(2) = Low(wheel)     _usb_tx_buffer2(3) = High(wheel)     _usb_tx_buffer2(4) = Low(buttons)     _usb_tx_buffer2(5) = High(buttons)     _usb_tx_buffer2(6) = Buffer_6     _usb_tx_buffer2(7) = Buffer_7     _usb_tx_buffer2(8) = Buffer_8     _usb_tx_buffer2(9) = Combined_pedals     Call Usb_send(_usb_tx_status2 , 8)

Программа закончена, можно ее скомпилировать и зашить в контроллер.

Скачиваем драйвер с сайта Logitech, устанавливаем его и подключаем наше устройство. Драйвер определит его как G25, заходим в свойства, здесь можно протестировать поворот руля, нажатие педалей — все сразу отразится на экране. Кроме того, нажимая на кнопки, можно протестировать FFB — «езда по деревянному мосту», «лопнувшая покрышка», «взрыв»…

 Второй руль, который мы будем создавать — Logitech Driving Force™ Pro:

Driving Force Pro.jpg

Это престижная модель с двумя педалями и переключателем передач на встроенной консоли. Передачи переключаются только «вперед/назад». На консоли руля больше кнопок, чем на G25, но их суммарное число меньше — 18. Такая конфигурация очень распространена среди рулей с FFB, поэтому прошивка под нее будет наиболее востребована для ремонта или апгрейда.

Схема остается прежняя, вместо отсутствующей педали сцепления подключим потенциометр чувствительности. С его помощью можно будет, не заходя в настройки, прямо во время игры подстроить силу воздействия FFB:

Schematic DFP.png

У нас освободились 11 бит, поэтому мы можем установить на педали разрядность в 8 бит, а на колесо руля — 14. Это повысило точность и плавность управления. Изменения в дескрипторе и программе здесь приводить не буду — их можно посмотреть в прилагаемых исходниках.

Теперь попробуем сделать геймпад. За основу возьмем Logitech Rumblepad 2:

rumblepad-1.jpg

Здесь FFB вибрационного типа, т.е. установлены два вибромотора, направление вращения которых не регулируется, только скорость. Поэтому вместо мостовой схемы применяются просто ключевые транзисторы и два выхода ШИМ. В ATmega8 два выхода имеет Таймер1, они уже выведены на разъем. Кроме того, для одномоторного случая можно задействовать освободившийся выход Таймера2 и подавать на него мажоритарный уровень левого и правого моторов FFB:

schematic Rumble.png

Теперь, не меняя схему и прошивку, можно подключить либо один, либо два мотора, в зависимости от конструкции устройства.

Конфигурируем Таймер1:

 Config Timer1 = Pwm , Pwm = 8 , Compare_a_pwm = Clear_up , Compare_b_pwm = Clear_up , Prescale = 8  Compare1a = 0  Compare1b = 0  Config Portb.1 = Output  Config Portb.2 = Output 

И Таймер2:

 Config Timer2 = Pwm , Prescale = 8 , Compare Pwm = Clear Up

Съем данных с потенциометров и кнопок аналогичен предыдущим. Репорт-дескриптор имеет незначительные изменения.

Пакет для отправки хосту имеет следующий вид:

     _usb_tx_buffer2(2) = Low(dpad_x)     _usb_tx_buffer2(3) = Low(dpad_y)     _usb_tx_buffer2(4) = Low(rjoy_x)     _usb_tx_buffer2(5) = Low(rjoy_x)     _usb_tx_buffer2(6) = Low(rjoy_y)     _usb_tx_buffer2(7) = Low(buttons)     _usb_tx_buffer2(8) = High(buttons)     _usb_tx_buffer2(9) = 64     Call Usb_send(_usb_tx_status2 , 8)

Последний байт всегда равен 64, таково требование драйвера — это своеобразный пароль.

Вот теперь, собственно, и все. По аналогии с вышеизложенным можно сделать джойстик, самолетный штурвал и проч. В приложении все схемы, пример платы в формате Cadsoft Eagle, исходники и откомпилированные коды. Библиотеку SWUSB.LBX перед компиляцией следует поместить в каталог Lib.

P.S. Использование драйвера не противозаконно, у меня были сомнения на этот счет, поэтому я испросил разрешение у фирмы Logitech и ответ был таков: мы не можем запретить Вам такое использование, однако, в этом случае мы не несем никакой ответственности за … и т.д.   

Последнюю версию драйвера можно скачать здесь.

Список радиоэлементов

Обозначение Тип Номинал Количество Примечание Магазин Мой блокнот
IC1 МК AVR 8-бит ATmega8 1
Q2, Q8 Биполярный транзистор BC807 2
Q3, Q5, Q6 MOSFET-транзистор IRF9540 3
Q4, Q7, Q9 Транзистор IRLML2402 3
Q10, Q11 MOSFET-транзистор IRF530 2
D1, D2 Стабилитрон 3.6 В 1
D4 Выпрямительный диод UF5404 1
D3, D5, D6, D7 Диод 4
С1-С7, C13 Конденсатор 0.1 мкФ 8
С8, С14, С16 Электролитический конденсатор 47 мкФ 3
С9, С10 Конденсатор 18 пФ 1
С11 Электролитический конденсатор 22 мкФ 1
С12 Электролитический конденсатор 1000 мкФ 35В 1
С15 Конденсатор 1 мкФ 1
R1, R14, R18, R34, R20, R29 Резистор 10 кОм 6
R3-R13 Резистор 470 Ом 10
R15 Резистор 1.5 кОм 1
R16, R17 Резистор 68 Ом 1
R18 Резистор 4.7 кОм 1
R22, R35 Резистор 10 Ом 2
R23 Резистор 2 кОм 1
R24, R31, R32, R33 Переменный резистор 4
R25, R26, R30 Резистор 270 Ом 3
R27, R19, R28, R21 Резистор 1 кОм 4
LED1 Светодиод 1
Q1 Кварцевый резонатор 12 МГц 1
L1 Катушка индуктивности 10 мГн 1
L2 Катушка индуктивности 22 мкГн 1