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/>.
28 #ifdef USE_MSP_DISPLAYPORT
32 #include "common/utils.h"
34 #include "drivers/display.h"
35 #include "drivers/osd.h"
37 #include "io/displayport_msp.h"
40 #include "msp/msp_protocol.h"
41 #include "msp/msp_serial.h"
47 static displayPort_t mspDisplayPort
;
48 static serialPortIdentifier_e displayPortSerial
;
50 static int output(displayPort_t
*displayPort
, uint8_t cmd
, uint8_t *buf
, int len
)
55 // FIXME There should be no dependency on the CLI but mspSerialPush doesn't check for cli mode, and can't because it also shouldn't have a dependency on the CLI.
60 return mspSerialPush(displayPortSerial
, cmd
, buf
, len
, MSP_DIRECTION_REPLY
, MSP_V1
);
63 static int heartbeat(displayPort_t
*displayPort
)
65 uint8_t subcmd
[] = { MSP_DP_HEARTBEAT
};
67 // heartbeat is used to:
68 // a) ensure display is not released by MW OSD software
69 // b) prevent OSD Slave boards from displaying a 'disconnected' status.
70 output(displayPort
, MSP_DISPLAYPORT
, subcmd
, sizeof(subcmd
));
75 static int grab(displayPort_t
*displayPort
)
77 return heartbeat(displayPort
);
80 static int release(displayPort_t
*displayPort
)
82 uint8_t subcmd
[] = { MSP_DP_RELEASE
};
84 return output(displayPort
, MSP_DISPLAYPORT
, subcmd
, sizeof(subcmd
));
87 static int clearScreen(displayPort_t
*displayPort
, displayClearOption_e options
)
91 uint8_t subcmd
[] = { MSP_DP_CLEAR_SCREEN
};
93 return output(displayPort
, MSP_DISPLAYPORT
, subcmd
, sizeof(subcmd
));
96 static bool drawScreen(displayPort_t
*displayPort
)
98 uint8_t subcmd
[] = { MSP_DP_DRAW_SCREEN
};
99 output(displayPort
, MSP_DISPLAYPORT
, subcmd
, sizeof(subcmd
));
104 static int screenSize(const displayPort_t
*displayPort
)
106 return displayPort
->rows
* displayPort
->cols
;
109 static int writeString(displayPort_t
*displayPort
, uint8_t col
, uint8_t row
, uint8_t attr
, const char *string
)
111 #define MSP_OSD_MAX_STRING_LENGTH 30 // FIXME move this
112 uint8_t buf
[MSP_OSD_MAX_STRING_LENGTH
+ 4];
114 int len
= strlen(string
);
115 if (len
>= MSP_OSD_MAX_STRING_LENGTH
) {
116 len
= MSP_OSD_MAX_STRING_LENGTH
;
119 buf
[0] = MSP_DP_WRITE_STRING
;
122 buf
[3] = displayPortProfileMsp()->fontSelection
[attr
& (DISPLAYPORT_SEVERITY_COUNT
- 1)] & DISPLAYPORT_MSP_ATTR_FONT
;
124 if (attr
& DISPLAYPORT_BLINK
) {
125 buf
[3] |= DISPLAYPORT_MSP_ATTR_BLINK
;
128 memcpy(&buf
[4], string
, len
);
130 return output(displayPort
, MSP_DISPLAYPORT
, buf
, len
+ 4);
133 static int writeSys(displayPort_t
*displayPort
, uint8_t col
, uint8_t row
, displayPortSystemElement_e systemElement
)
137 syscmd
[0] = MSP_DP_SYS
;
140 syscmd
[3] = systemElement
;
142 return output(displayPort
, MSP_DISPLAYPORT
, syscmd
, sizeof(syscmd
));
145 static int writeChar(displayPort_t
*displayPort
, uint8_t col
, uint8_t row
, uint8_t attr
, uint8_t c
)
151 return writeString(displayPort
, col
, row
, attr
, buf
);
154 static bool isTransferInProgress(const displayPort_t
*displayPort
)
160 static bool isSynced(const displayPort_t
*displayPort
)
166 static void redraw(displayPort_t
*displayPort
)
168 drawScreen(displayPort
);
171 static uint32_t txBytesFree(const displayPort_t
*displayPort
)
174 return mspSerialTxBytesFree();
177 static const displayPortVTable_t mspDisplayPortVTable
= {
180 .clearScreen
= clearScreen
,
181 .drawScreen
= drawScreen
,
182 .screenSize
= screenSize
,
183 .writeSys
= writeSys
,
184 .writeString
= writeString
,
185 .writeChar
= writeChar
,
186 .isTransferInProgress
= isTransferInProgress
,
187 .heartbeat
= heartbeat
,
189 .isSynced
= isSynced
,
190 .txBytesFree
= txBytesFree
,
191 .layerSupported
= NULL
,
196 displayPort_t
*displayPortMspInit(void)
198 displayInit(&mspDisplayPort
, &mspDisplayPortVTable
, DISPLAYPORT_DEVICE_TYPE_MSP
);
200 if (displayPortProfileMsp()->useDeviceBlink
) {
201 mspDisplayPort
.useDeviceBlink
= true;
205 if (vcdProfile()->video_system
!= VIDEO_SYSTEM_HD
) {
206 vcdProfileMutable()->video_system
= VIDEO_SYSTEM_HD
;
210 if (vcdProfile()->video_system
== VIDEO_SYSTEM_HD
) {
211 vcdProfileMutable()->video_system
= VIDEO_SYSTEM_AUTO
;
215 if (vcdProfile()->video_system
== VIDEO_SYSTEM_HD
) {
216 mspDisplayPort
.rows
= osdConfig()->canvas_rows
;
217 mspDisplayPort
.cols
= osdConfig()->canvas_cols
;
219 const uint8_t displayRows
= (vcdProfile()->video_system
== VIDEO_SYSTEM_PAL
) ? VIDEO_LINES_PAL
: VIDEO_LINES_NTSC
;
220 mspDisplayPort
.rows
= displayRows
+ displayPortProfileMsp()->rowAdjust
;
221 mspDisplayPort
.cols
= OSD_SD_COLS
+ displayPortProfileMsp()->colAdjust
;
224 redraw(&mspDisplayPort
);
226 return &mspDisplayPort
;
229 void displayPortMspSetSerial(serialPortIdentifier_e serialPort
) {
230 displayPortSerial
= serialPort
;
232 #endif // USE_MSP_DISPLAYPORT