2 * This file is part of Cleanflight and Betaflight.
4 * Cleanflight and Betaflight are free software. You can redistribute
5 * this software and/or modify this software under the terms of the
6 * GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option)
10 * Cleanflight and Betaflight are distributed in the hope that they
11 * will be useful, but WITHOUT ANY WARRANTY; without even the implied
12 * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 * See the GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this software.
18 * If not, see <http://www.gnu.org/licenses/>.
23 * http://fpv-treff.de/viewtopic.php?f=18&t=1368&start=3020#p44535
24 * http://fpv-community.de/showthread.php?29033-MultiWii-mit-Graupner-SUMD-SUMH-und-USB-Joystick-auf-ProMicro
33 #ifdef USE_SERIALRX_SUMH
35 #include "common/utils.h"
37 #include "drivers/time.h"
39 #include "io/serial.h"
42 #include "telemetry/telemetry.h"
50 #define SUMH_BAUDRATE 115200
52 #define SUMH_MAX_CHANNEL_COUNT 8
53 #define SUMH_FRAME_SIZE 21
55 static bool sumhFrameDone
= false;
57 static uint8_t sumhFrame
[SUMH_FRAME_SIZE
];
58 static uint32_t sumhChannels
[SUMH_MAX_CHANNEL_COUNT
];
60 static serialPort_t
*sumhPort
;
62 // Receive ISR callback
63 static void sumhDataReceive(uint16_t c
, void *data
)
68 static uint32_t sumhTimeLast
, sumhTimeInterval
;
69 static uint8_t sumhFramePosition
;
72 sumhTimeInterval
= sumhTime
- sumhTimeLast
;
73 sumhTimeLast
= sumhTime
;
74 if (sumhTimeInterval
> 5000) {
75 sumhFramePosition
= 0;
78 sumhFrame
[sumhFramePosition
] = (uint8_t) c
;
79 if (sumhFramePosition
== SUMH_FRAME_SIZE
- 1) {
80 // FIXME at this point the value of 'c' is unused and un tested, what should it be, is it important?
87 static uint8_t sumhFrameStatus(rxRuntimeState_t
*rxRuntimeState
)
89 UNUSED(rxRuntimeState
);
94 return RX_FRAME_PENDING
;
97 sumhFrameDone
= false;
99 if (!((sumhFrame
[0] == 0xA8) && (sumhFrame
[SUMH_FRAME_SIZE
- 2] == 0))) {
100 return RX_FRAME_PENDING
;
103 for (channelIndex
= 0; channelIndex
< SUMH_MAX_CHANNEL_COUNT
; channelIndex
++) {
104 sumhChannels
[channelIndex
] = (((uint32_t)(sumhFrame
[(channelIndex
<< 1) + 3]) << 8)
105 + sumhFrame
[(channelIndex
<< 1) + 4]) / 6.4f
- 375;
107 return RX_FRAME_COMPLETE
;
110 static float sumhReadRawRC(const rxRuntimeState_t
*rxRuntimeState
, uint8_t chan
)
112 UNUSED(rxRuntimeState
);
114 if (chan
>= SUMH_MAX_CHANNEL_COUNT
) {
118 return sumhChannels
[chan
];
121 bool sumhInit(const rxConfig_t
*rxConfig
, rxRuntimeState_t
*rxRuntimeState
)
125 rxRuntimeState
->channelCount
= SUMH_MAX_CHANNEL_COUNT
;
126 rxRuntimeState
->rcReadRawFn
= sumhReadRawRC
;
127 rxRuntimeState
->rcFrameStatusFn
= sumhFrameStatus
;
129 const serialPortConfig_t
*portConfig
= findSerialPortConfig(FUNCTION_RX_SERIAL
);
135 bool portShared
= telemetryCheckRxPortShared(portConfig
, rxRuntimeState
->serialrxProvider
);
137 bool portShared
= false;
140 sumhPort
= openSerialPort(portConfig
->identifier
, FUNCTION_RX_SERIAL
, sumhDataReceive
, NULL
, SUMH_BAUDRATE
, portShared
? MODE_RXTX
: MODE_RX
, (rxConfig
->serialrx_inverted
? SERIAL_INVERTED
: 0));
144 telemetrySharedPort
= sumhPort
;
148 return sumhPort
!= NULL
;