The SPRacing Pixel OSD system is a system that allows overlaying graphics, including dots; lines; shapes; text onto an analog video signal (PAL or NTSC). It also features signal generation code to generate the video signal sync pulses.
Example OSD overlay, showing pixels and text:
- SPRacingH7RF - 2020/05/16 - https://www.youtube.com/watch?v=yxXjFuN0uPc
- SPRacingH7CINE - 2020/03/04 - https://www.youtube.com/watch?v=1IUbb08_iKM
Here you can see the typical PCB footprint, on the SPRacingH7CINE (Left), SPRacingH7RF (Center), SPRacingH7CL (Right)
Dominic Clifton created every part of the system, including design, architecture, schematics, prototypes and production hardware.
- Very low-cost!
- Framebuffer based.
- Single or double buffered.
- 2 bits per pixel, black, white, grey, transparent.
- Adjustable resolution based on available framebuffer RAM and timer settings for pixel-clock.
- Very low CPU overhead, pixel stream is DMA based, CPU needs to respond promptly to video sync ISRs and transfer-complete ISRs.
- RAM for framebuffer. Actual amount of RAM depends on how many lines of pixels you want to display and how big the pixels are.
- Hardware timers for Sync, Pixel, hardware timers need to be linked to the DAC and COMP and each other.
- Single DAC channel (2 DAC channels can be used for debugging purposes).
- Comparator.
- Clock speed fast enough to handle drawing operations, sync and pixel clocks.
- A GPIO port with 4 pins connected. No other pins on the port can be configured to be DIGITAL OUT. Pins on the same port CAN be used in DIGITAL IN, Alternate Function, Etc. This is because the GPIO's ODR register is used.
The system works on many STMicro H7 CPUs and the L4 CPUs. Others can be used.
The STM32L432KC was uses for prototyping as is very low cost and comes in extremely small 32-pin packages, has necessary peripherals, an 80Mhz clock speed and enough RAM to support a good-resolution framebuffer.
STM32L432KC - https://www.st.com/en/microcontrollers-microprocessors/stm32l432kc.html
STM32H730VB - https://www.st.com/en/microcontrollers-microprocessors/stm32h730-value-line.html
For the H723/H730/H733/H743/H750 the following peripherals are used.
- DMA2 (2 streams, Stream 6 + 7). 1 stream for sync pulse generation, 1 stream for pixel generation.
- MDMA - for DMA based frame-buffer clearing.
- DAC1 - for generation of the voltage used by the comparator for video sync.
- COMP2 - for detecting sync pulses.
- ADC1 - for sampling video volages.
- TIM1 - Sync and video generation.
- TIM2 - ADC sampling.
- TIM15 - Pixel generation timer.
- GPIOE - Pixel generation IO.
- Fast Analog switch with negative voltage handling.
- Fast signal switching diode.
- Video filter + passives for sync-pulse noise rejection.
- ~12 passive components (resistors, capacitors).
The SPRacingPixelOSD system is available in these products, this list will be updated as more are made.
Manufacturer | Product | Description | Website |
---|---|---|---|
SPRacing | SPRacingH7RF | H730 based flight controller | http://seriouslypro.com/spracingh7rf |
FlightOne * | LightningH7 | H730 based flight controller |
- Do not order anything from the FlightOne website, they are not currently fulfilling any orders.
Entity | Project | Description | Website |
---|---|---|---|
SPRacing | SPRacingH7CINE | H750 based flight controller | N/A |
SPRacing | SPRacingH7CL | H723/H733 based flight controller | N/A |
For a long time the author has been wanting to push forward the technology used in flight-controller hardware and software. FPV drones generally all use the same OSD chip, a MAX7456, which has long been discontinued by Maxim and is only available from China-clone manufacturers under the AT7456/E.
The MAX7456 chip has major limitations:
- Obsolete Product Lifecycle status - https://www.analog.com/en/products/max7456.html
- Only supports characters.
- Is only available in a very large form factor.
- Is expensive, and contributes massively to the overall BOM cost.
- Has a 'large' PCB footprint for a 20x20mm flight controller.
- Limits software development which is stuck in the 1970's era of text based overlays where only text and crude text-based images can be displayed (think Teletext/Ceefax).
To reduce BOM cost, and add features and reduce overall PCB footprint, the PixelOSD system can use the same processor that runs the flight code. An example of this can be found in the SPRacingH7RF which uses an STM32H730VBH6 MCU to run flight code, Pixel OSD code and SPI ExpressLRS RF radio receiver code - all in a 20x20 mount PCB!
The original prototype and code was created in April 2019.
The schematics, documentation and code are available under the Apache License, content of the NOTICE file, repeated here below, needs to be included wherever schematics, code and documentation are used or in any system that displays text via the OSD system. E.g. in standalone OSD systems, or when used by flight-control systems, e.g via way of a NOTICES menu in a menu system, or a briefly displayed boot screen.
The NOTICE also needs to be included as a note on any schematics (e.g. PDFs, images) using schematics derived from this project. Please ensure that you add a text element in your EDA software containing the NOTICE.
If the above displaying of the NOTICE file isn't technically possible then please contact Dominic Clifton to discuss potential solutions.
SPRacingPixelOSD pixel-based on-screen-display system.
Copyright 2019-2023 Dominic Clifton.
This product includes software and hardware designs developed by Dominic Clifton.
* Dominic Clifton - http://dominicclifton.name
* Seriously Pro / SP Racing - http://seriouslypro.com
The Initial Developer of the SPRacingPixelOSD system is Dominic Clifton.
Copyright 2019 - 2023 Dominic Clifton. All Rights Reserved.
As noted below, the code in the prototype
folder contains code from other projects, which are not licensed under
the same terms as above and thus must NOT be used for production code as the licenses conflict with each other.
Many thanks to the respective authors, listed in the 'Acknowledgements and references' later in this document.
A video signal is an analog signal.
It has two main parts.
- Sync pulses to indicate start of field.
- per field Horizontal line signals.
The sync pulses are output at the start of a frame, and are per-video system (e.g. PAL/NTSC). PAL/NTSC can be detected by counting the frame sync pulse lengths and the length of them.
The horizontal line signals comprises of a low voltage sync pulse, a black-level signal, a color bust, black level again, then a per-line voltages, then black level again.
Some cameras output negative sync pulse voltages. The sync pulse voltage can change over time, e.g. as a camera warms up or if the camera's average voltage level changes.
This system can handle both of the above. However, most modern cameras output a 0-0.3 volt stable sync-voltage.
You can read more about the PAL/NTSC video signal specifications here:
In order to detect sync, the system uses the following techniques.
-
Low-cost hardware Video filter to eliminate noise and boost the output voltage for wider voltage thresholds.
A FMS6141 was selected.
-
A hardware RC low-pass filter comprised of a resistor and capacitor.
Formula: 1/2pi(RF)(CF) = 1/2pi(100)(570) = 2.79Mhz
100R and 560pF was used to get pretty close using standard values. IMPORTANT: Use high-tolerance parts!
Refer to Renesas AN1269/AN1316 - "One Transistor Enables Clean HDTV and NTSC Video Sync Separation" See
application-notes
folder in this repository. -
A fast comparator (usually internal to the MCU) that generates an ISR for both signal edge pulses.
-
A DAC to generate a reference voltage for the comparator. No external pin is required when using a comparator in the MCU as it is routed internally on the MCU, but the voltage can also be exposed on a pin, for debugging purposes.
-
A timer which is linked to the comparator edge transition signal, to record the sync pulse length in hardware via a capture compare channel.
-
Gating of the comparator output signal for the horizontal field portion of the signal, so that voltages close to the comparator threshold voltage do not trigger the timer or additional ISRs. This is technique is very important and is achieved by using an additional channel on the comparator-transition-linked timer which is internally linked to the comparator's blanking input.
-
Counting of short-vs-long sync pulses to detect the start of the frame and adjust the DAC output voltage. That is the comparator voltage is either two high or two low if there are two few or two many incorrect-length pulses.
Video sync detection needs to be quick so that boot logos of flight control software can be displayed.
A second timer is started automatically by the sync-detection timer in hardware so that it always starts the correct duration after the sync pulses. It is critical that it's done in hardware so that pixels generated on the first horizontal-line line-up with the pixels on subsequent lines. The second timer cannot be started by software, thus it is a hardware requirement that the pixel generation timer has both DMA update and is internally linked in the MCU to the video sync timer. On the L4 and H7 this means using TIM1 (sync) and TIM15 (pixel).
The timer is not started immediately, but needs to be started a short time after the sync pulse, thus another channel on the sync timer is used as a delay timer, which when the delay expires triggers the pixel timer which runs until it is stopped.
A DMA stream is configured, the source address is the line-buffer, the destination address is the ODR register of a GPIO port.
The framebuffer contains the 2 bits per pixel image information.
Before a line of pixels can be output, a line-buffer needs to be created from the pixel-buffer. This is achieved by mapping the bits that indicate the various pixel states to the corresponding bits that toggle GPIO pin output line levels when the ODR is written to.
The pixel timer frequency is configured so that one DMA update event is triggered for each pixel. When the DMA update event is generated the DMA controller copies one element from the line buffer to the GPIO ODR register.
In order to reduce DMA bandwidth only a single byte is transferred from the line buffer to the ODR register, but the ODR register is a 16 bit register. So either the high 8 bits of the ODR are used, or the low 8 bits are used. That in turn means that all the IO for the pixel generation circuit needs to be on the first 8 pins of the GPIO port or the last 8 pins of the GPIO port. e.g. 0-7 or 8-15, not a mix of pins from 0-15.
The pixel timer is stopped by the DMA transfer-complete ISR.
When a sync pulse is generated, the software creates the line buffer from the frame buffer, this needs to be quick and is done in the comparator interrupt handler.
Given that the timer for pixel generation is started by the timer linked to the comparator it's only possible to output pixels when video sync is detected and present.
The system has the ability to generate a video signal with no pixels (i.e. just the field and line sync pulses).
The sync signal is generated by using another channel on the same timer that is usually used for sync detection. a PWM signal is used, the lengths of the pulses are transferred by DMA to the timer's registers, three registers are used for this: ARR, REPETITION and CC1.
The sync generation DMA stream is configured in burst mode so that when a DMA update event occurs three items are transferred to the timer registers in one go via BURST DMA. Note that ARR, REPETITION and CC1 have addresses that form a contiguous space on both the DMA stream buffer and on the timer's peripheral memory space. This is why only Channel 1 can be used for sync generation, this in turn means that the timer's CC1 output pin is also the pin used in the schematic.
The lengths of the pulses for the PWM signal are pre-calculated by the software at compile-time and two DMA buffers are created, one for PAL and one for NTSC.
If a camera sync signal is present when also generating sync signals then the sync will be bad and the pixel clock will be started at the wrong time resulting in display corruption.
Video sync output is only started if sync detection fails.
IMPORTANT: Please read the descriptions above prior to reading this section, otherwise it won't make sense.
The analog switch has two independent channels:
- Used for selection of white pixel voltage.
- Used for masking of the incoming video signal.
The OSD_MASK_EN disconnects the video input signal voltage from the voltages generated by other components connected to the VIDEO_IN_MASK and shunts the VIDEO_IN signal to GND via a 75ohm termination resistor.
The WHITE_SOURCE_FIXED signal is a MCU GPIO level voltage, 3.3v, for a fixed-voltage source for white pixels.
The MCU generates a voltage on the DAC1_OUT/WHITE_SOURCE DAC channel. Between 0 and 3.3v.
The WHITE_SOURCE_SELECT signal either connects WHITE_SOURCE_FIXED or DAC1_OUT to the WHITE_SOURCE net.
The output of Analog Switch channel 2 is WHITE_SOURCE.
The WHITE_SOURCE net then goes through a one half of a pair of voltage divider resistors, the net is called WHITE_SOURCE_OUT.
The VIDEO_SYNC_OUT also goes through one half of a voltage divider resistor, the output net is called VIDEO_SYNC.
Then VIDEO_SYNC and WHITE_SOURCE_OUT are mixed via a fast signal switching diode in dual anode configuration.
The diode is there to protect the MCU and to further drop the voltage of the DAC1_OUT, VIDEO_SYNC_OUT and WHITE_SOURCE_FIXED signals.
The voltage divider circuit is formed of 4 resistors:
First half (before the diode):
- WHITE_SOURCE -> WHITE_SOURCE_OUT
- VIDEO_SYNC_OUT -> VIDEO_SYNC
Second half (after the diode): 3. 75ohm resistor to GND. 4. 75ohm resistor to GPIO PIN in Open-Drain mode, this is connected to BLACK_SINK_1 (Yes, SINK, not SYNC).
When BLACK_SINK_1 is enabled, the second half of the voltage divider is then comprised of two resistors in parallel which will change how the voltage is divided.
Thus, the voltage on the VIDEO_IN_MASKED can be increased or decreased by the MCU's GPIO and DAC outputs.
Given that the GPIO port has 4 wires connected to it and that VIDEO_SYNC_OUT is combined, there are only certain valid combinations of IO levels that make any sense.
This table will be updated, as time permits, with the valid combinations. For now refer to the source-code where the line-buffer is generated from the pixel buffer.
Ref. | VSO | WSF | WSS | OME | BS1 | Valid? | Effect |
---|---|---|---|---|---|---|---|
0 | 0 | 0 | 0 | 0 | 0 | TODO | TODO |
1 | 0 | 0 | 0 | 0 | 1 | TODO | TODO |
2 | 0 | 0 | 0 | 1 | 0 | TODO | TODO |
3 | 0 | 0 | 0 | 1 | 1 | TODO | TODO |
4 | 0 | 0 | 1 | 0 | 0 | TODO | TODO |
5 | 0 | 0 | 1 | 0 | 1 | TODO | TODO |
6 | 0 | 0 | 1 | 1 | 0 | TODO | TODO |
7 | 0 | 0 | 1 | 1 | 1 | TODO | TODO |
8 | 0 | 1 | 0 | 0 | 0 | TODO | TODO |
9 | 0 | 1 | 0 | 0 | 1 | TODO | TODO |
10 | 0 | 1 | 0 | 1 | 0 | TODO | TODO |
11 | 0 | 1 | 0 | 1 | 1 | TODO | TODO |
12 | 0 | 1 | 1 | 0 | 0 | TODO | TODO |
13 | 0 | 1 | 1 | 0 | 1 | TODO | TODO |
14 | 0 | 1 | 1 | 1 | 0 | TODO | TODO |
15 | 0 | 1 | 1 | 1 | 1 | TODO | TODO |
16 | 1 | 0 | 0 | 0 | 0 | TODO | TODO |
17 | 1 | 0 | 0 | 0 | 1 | TODO | TODO |
18 | 1 | 0 | 0 | 1 | 0 | TODO | TODO |
19 | 1 | 0 | 0 | 1 | 1 | TODO | TODO |
20 | 1 | 0 | 1 | 0 | 0 | TODO | TODO |
21 | 1 | 0 | 1 | 0 | 1 | TODO | TODO |
22 | 1 | 0 | 1 | 1 | 0 | TODO | TODO |
23 | 1 | 0 | 1 | 1 | 1 | TODO | TODO |
24 | 1 | 1 | 0 | 0 | 0 | TODO | TODO |
25 | 1 | 1 | 0 | 0 | 1 | TODO | TODO |
26 | 1 | 1 | 0 | 1 | 0 | TODO | TODO |
27 | 1 | 1 | 0 | 1 | 1 | TODO | TODO |
28 | 1 | 1 | 1 | 0 | 0 | TODO | TODO |
29 | 1 | 1 | 1 | 0 | 1 | TODO | TODO |
30 | 1 | 1 | 1 | 1 | 0 | TODO | TODO |
31 | 1 | 1 | 1 | 1 | 1 | TODO | TODO |
Care must be taken not to enable/generate PWM signals on the VIDEO_SYNC_OUT pin at the wrong time, and that the timer's idle levels are correct, and that too high a voltage is not given to the video filter.
In practice though it turned out that you either get no output or the wrong output at the wrong time and no damage to the MCU, camera, video filter has ever been done even after trying all the above combinations at one point or another during development.
The VIDEO_IN_MASKED is then fed through a video filter. The video filter is there for 3 primary reasons.
- Boost the video voltages somewhat, 2x + 150mv DC offset. This 150mv offset helps bring up the sync level voltage to a voltage that is better for the comparator and DAC to use for detection, and to allow for selection of appropriate high/low threshold voltages.
- Cleanup any noise in the signal.
- Allow successful driving of various video devices (vtx, screen, capture card, etc) irrespective of the voltages at the input to the filter.
The output of the filter, VIDEO_FILTER_OUT is then used as follows:
- Via a 75R resistor to video-out, a 75R series termination resistor is used to reduce the video output voltage level back to normal levels, as the video filter doubles the voltage. See video filter datasheet. "Output considerations".
- Via the RC filter that filters out the color bust signal the output is then fed to the comparator on the MCU as VIDEO_SYNC_IN.
The rest of the video circuit is then made up by the GPIO, Comparator and Timer (TIM1) PWM generation and / CC circuits inside the MCU.
There are also two connections to ADC1 for VIDEO_IN and VIDEO_OUT, these can be used to see if a camera is connected, measure sync pulse or black-level voltages and check video output generation.
All components, in the reference schematic can of course be changed, optimized or substituted at will.
The reference design is optimized for the lowest BOM count, using the cheapest and most-available parts at the time. Over time availability and prices change, so feel free to do what is necessary for your project when using the schematics.
However, be aware that:
- The resistor values are all specific to the diode and video filter. Changing the diode will require re-verification of voltages and video signal pixel generation timing/quality.
- The resistor and capacitor values on the RC sync filter are CRITICAL.
- The analog switch was the fastest that was available at the time, and handles negative voltages. Using a slower one will reduce or break pixel generation timing/quality. Using one without negative voltage handling limits compatability to cameras with a 0v or positive sync voltage only. Be aware that older video equipment uses a ~ -0.3v sync voltage, and 0v black level.
- MCU can be changed, but check peripheral interconnect matrix and finer details of peripheral interconnect signals that your chosen MCU has.
- H730 is a really-good choice for any non-flight control application as it cheap and handles it with ease.
- L4 is a great choice for basic integration with a VTX for example, with FC->VTX communication over a UART.
- If you want a single MCU solution for FC+OSD, with no external flash, then the H723/H733 are great.
- If you want the cheapest possible FC+OSD solution, and need a lot of flash, then the H730 is great, and has the fastest clock speed at 520-550Mhz.
- If you want a cheap FC+OSD solution and need a lot of RAM, then the H750 is great, but slower than the H723/730/733.
- If you want a good all round FC+OSD solution, with lots of flash and RAM then the H743/H753 are good candidates as they have more RAM and flash than the H723/H730/H733 but run slower at 480Mhz.
- A CPU with support for memory mapped flash is good if you want lots of static graphics/artwork, as you can use a QUADSPI or OCTOSPI flash and write code to copy directly from external flash into the framebuffer. Probably also via DMA too.
Missing an interrupt is BAD. Handling an interrupt late is BAD.
If ISRs are late or missing the following can occur:
- Loss-of-sync.
- Bad overlay output.
- Black level held to long at end-of-frame which corrupts the sync pulse.
When integrating the OSD into flight controller software note that video ISR's need to be the highest priority ISRs in the system. For a flight controller this means that it's more important to handle the OSD than the GYRO, there's no point handling a gyro EXTI if you can't see where you're going when flying FPV due to loss-of-sync! One slightly late handling of a gyro EXTI signal won't cause noticeable flight behavior.
The software was originally written to be a stand-alone system that runs on an STM32L432KC and was developed using the NucleoL432KC development board.
The system was then later directly integrated into a popular flight control system, but was never released in this form.
However, for the curious, here's a branch with it:
Refer to the following directories:
- https://github.com/spracing/betaflight/blob/bf-spracingh7cine-target-20200304-before-osdlib/src/main/target/SPRACINGH7CINE
- https://github.com/spracing/betaflight/tree/bf-spracingh7cine-target-20200304-before-osdlib/src/main/drivers/spracingpixelosd
- https://github.com/spracing/betaflight/tree/bf-spracingh7cine-target-20200304-before-osdlib/src/main/osd
- https://github.com/spracing/betaflight/blob/bf-spracingh7cine-target-20200304-before-osdlib/src/main/io/displayport_spracing_pixel_osd.c
Later, the video sync and video generation code was moved into a library for a number of reasons:
- re-use.
- maintainability.
- allows use with code compiled using a different compiler, as long as the ABI is common.
- cannot be broken by changes to the flight-controller code.
- allows the OSD bios code to be updated and released on a different schedule to the flight-controller code.
The library is in the library
folder of this repository.
The library uses two frame-buffers, and this doubles the frame-buffer memory requirement.
Two frame-buffers are used so that the flight control software can draw into one frame-buffer, when it has time, whilst the video overlay is being displayed from the other frame-buffer.
If the flight control software's scheduler is good enough, you can achieve 50(PAL)/60(NTSC) frames-per-second which looks amazingly smooth. If the flight controller's scheduler doesn't devote much time to rendering, or the rendering code is slow then the video overlay can be updated slower than 50/60 FPS but results in a suboptimal user experience.
The library is compiled on a per-target basis, as the pin, hardware and memory layout requirements differ per-target.
Currently, the build system for the library is designed to generate a binary which can be flashed to a target's external flash. The binary includes the hash for an EXST bootloader so that the bootloader can verify the binary is not corrupted.
The per-target pins, flash address and memory space is configured via the target's files. See the linker script .ld
and make .mk
and header .h
files in source\library\src\main\<TARGET>\*
For integration into flight control system, the flight controller linker scripts needs to reserve some RAM used by the library. An example of this can be found here:
The flight control system also needs to verify the presence of the library at the configured memory addresses. An example of this is here:
The flight controller needs to ensure that it reserves hardware resources that might otherwise be used by it, and then call initialisation functions in the library. An example of this is here:
The flight control system also needs to ensure that interrupts for the OSD system are routed to the library.
And example of this can be found here:
The flight control system then needs to draw to the frame-buffer and handle the onVSync
callback from the library.
The onVSync
handler can be used so the flight controller knows which of the two frame buffers to draw into and which
is being used to display the video overlay.
Some example frame-buffer rendering code can be found here:
Some example high-level drawing routine implementation can be found here:
The library can be compiled so that it can be:
- stored in MCU internal flash (e.g. STM32H743).
- stored on external flash and copied to RAM and run from RAM (e.g. STM32H750).
- stored on external flash and run from external flash via memory mapped support (e.g. STM32H730 via OCTOSPI).
- copied to RAM via a debugger and run from RAM.
All the above scenarios were performed during development of the library.
$ cd source/library
$ make TARGET=SPRACINGH7RF
Artifacts appear in the build
folder.
The following artifacts are generated:
- ELF file.
- ELF file patched for EXST bootloaders.
- Binary, suitable for flashing to an MCU.
- MAP file, useful for verification of configured memory addresses when integrating with FC build systems.
Example script for building and flashing via DFU can be found in library\support\scripts\build_and_flash.sh
.
e.g.
$ ./supports/scripts/build_and_flash.sh SPRACINGH7RF
The library can be compiled with debug symbols and source-level debugged using GDB, just load the symbols from the
.elf
artifact.
$ make DEBUG=1 TARGET=<TARGET> DEVELOPER_BUILD=YES
Then flash the /library/build/<TARGET>_DEVELOPER_DEBUG.bin
to the FC via DFU.
Then debug your application (e.g. flight control software) in the usual way, ensuring that you issue the following GDB command to load the symbols.
add-symbol-file /library/build/<TARGET>_DEVELOPER_DEBUG.elf
The original prototype code, which was unmaintained after the initial development can be found in the prototype
folder.
WARNING! Do NOT use the code from the prototype
folder other than for learning/education purposes. i.e. sure, run
it if you like, but for anything serious, or commercial, use the code in the library
folder.
Furthermore, the code in the prototype contains code from other projects, which are not licensed under the same terms as the library code and thus must not be used for production code as the licenses conflict with each other.
The code in the library
folder, when compared to the prototype
code has:
- improved video sync detection.
- improved video sync stability in noisy environments. (e.g. when used on a quad with more electrical noise).
- better camera compatibility.
- better video output signals.
- numerous additions.
- support for the analog switch.
- the prototype cannot generate black pixels, the overlay appears like a watermark.
- more organized code; the prototype code was quicker and dirtier and is proof-of-concept code.
- bug fixes.
PR's that cover the following are sought:
- Updated STM32L432 example, that uses the library code, included directly or as two distinct binaries.
- Runnable examples for other STM32 MCUs, such as H743.
- Corrections.
- Additional documentation.
- Improvements, especially performance or video sync related improvements.
There are two schematics, neither have been recently updated and this repository has been created at least two years after they were created so there might be minor changes/corrections needed.
There is a schematic for the prototype using an STM32L432KC, available here, in PDF, PNG and DipTrace formats:
See here: https://github.com/spracing/spracingpixelosd/tree/main/schematics/spracingpixelosd/prototype
There is a reference schematic for an STM32H7, available here, in PDF, PNG and DipTrace formats:
See here: https://github.com/spracing/spracingpixelosd/tree/main/schematics/spracingpixelosd/reference
- The sync stability could be improved, by using the comparator-timer-triggered-delayed ADC conversion to periodically sample the sync-voltage and dynamically adjust the thresholds and DAC voltage. On production boards this wasn't found to be necessary but some code that initialises the ADC was written for this purpose. This is also why ADC1 is reserved for video, so DO NOT use TIM2 and ADC1 in flight-controller code.
- Speed-up line-buffer generation.
- Use 4 more IO lines to generate different shades of grey, no extra DMA bandwidth required, but the framebuffer memory requirement would double. Would use more CPU time to generate line buffers, adds additional load to flight controller rendering code (e.g. 4-8 bits per pixel instead of 2).
At the time this document was written (2023/02/23) there are no known framebuffer-based open-source royalty free analog OSD systems.
- Vortex - The vortex OSD is great, but closed source and long discontinued, no published code or schematics.
- F1shpepper Tiny OSD - Isn't frame-buffer based. Doesn't integrate well into FC systems, no published schematics, not used in commercial projects.
- FrSkyOSD - More grey-scales, uses a resistor ladder, requires purchase from FrSky, high BOM cost, similar overall PCB footprint, no published code or schematics. Uses a UART for FC->OSD communication, no direct frame-buffer access.
- BrainFPV - FPGA based, no published code or schematics.
- Kiss Ultra - Single MCU solution, not reverse engineered. no published code or schematics. Probably works fairly similarly to this project, based on a quick look at photos of the PCB.
- SuperOSD - Uses two PIC processors, low resolution, old, unmaintained, archived.
Dominic Clifton has wanted to improve the state of OSD systems used in drones since way back when everyone was using the Arduino MAX7456 based MininOSD system with MultiWII way back in 2014.
- 2010 - SuperOSD system by Thomas Oldbury created.
- 2013/2014 - MinimOSD system first used by Dominic with MultiWII.
- 2015 - Dominic designed and shipped the SPRacingF3 flight controller under the SPRacing brand.
- 2015 - Dominic started designing the SPRacingOSD add-on board for the SPRacingF3. The idea was to create an OSD to replace the MinimOSD and then later transitiion the SPRacingOSD to a pixel based OSD. The latter never happened due to the amount of work required on the flight controller and Cleanflight projects.
- 2016 - SPRacingOSD add-on board shipped.
- 2016 - Immersion RC Vortex 285 shipped with Pixel based OSD, but limited OSD/FC integration with Cleanflight.
- 2017 - Fishpepper posts details and code for his pixel-based, but not framebuffer based, tinyOSD.
- 2019-Q1 - INAV OSD repo created, schematic used an STM32F103. Dominic contacted Fiam and initially had some technical discussions, but no schematics or running code were shared between either parties. Fiam was then offered a job by FrSky and all communication ceased due to his employment terms with FrSky. Fiam wanted a PixelOSD system that can be used by any flight controller, with a UART based protocol. Dominic, still wanting a pixel OSD system then spent a week designing a schematic and a week writing code and got pixels displayed on the screen, see photos folder. Note that it's no-coincidence the original SPRacingPixelOSD prototype and the final FrSkyOSD production module both use STM32L43x MCUs due to the technical discussions. Likely some use of the STM processor hardware features are used by both projects in the same way, but there's no confirmation of this.
- 2019-Q2 - SPRacingPixelOSD ported to H7.
- 2019-Q3 - SPRacingPixelOSD code integrated into Betaflight.
- 2019-Q4 - FlightOne interested in PixelOSD system, Dominic designed the LightningH7 Flight Controller for FlightOne. SPRacingPixelOSD code moved into a library, to be used by both Betaflight and FlightOne and Dominic integrated into both.
- 2020-Q1 - FlightOne shipped the LightningH7 FC with the SPRacingPixelOSD library.
- 2020-Q2 - FrSky shipped the FrSkyOSD.
- 2021-Q1 - Dominic designed the SPRacingH7RF FC with ELRS SPI, SPRacingPixelOSD, STM32H730 MCU and OctoSPI memory mapped flash.
- 2021-Q4 - SPRacing shipped the SPRacingH7RF.
- Thomas Oldbury - SuperOSD system. https://code.google.com/archive/p/super-osd/
- MWOSD - MultiWII OSD. http://www.mwosd.com/
- F1shpepper - TinyOSD video timing calculation. https://fishpepper.de/2019/03/11/tinyosd-tinyfinity-a-tiny-opensource-video-tx-with-full-graphic-osd/
- Fiam - Prototype MCU selection ideas and technical discussions - https://github.com/fiam / https://www.frsky-rc.com/product/frsky-osd/
- Maxim - Datasheets and application notes.
- Renasas - Datasheets and application notes.
- ImmersionRC - A vortex OSD schematics was reverse engineered before this project was completed, included in schematics, some technical details were investigated but discarded. https://www.immersionrc.com/fpv-products/vortex-racing-quad/
- FrSky - An FrSky OSD schematic was reverse engineered /after/ this project was completed, included in schematics.
- SPRacing Customers - For being awesome and supporting this work though purchases and feedback.
Likely there's some errors and omissions, as they are discovered they will be corrected. Please raise pull-requests via github to submit any corrections.