2013-04-20 18:07:19 +03:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* @addtogroup OpenPilotModules OpenPilot Modules
|
|
|
|
* @{
|
|
|
|
* @addtogroup OSDgenModule osdgen Module
|
|
|
|
* @brief Process OSD information
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file osdgen.h
|
|
|
|
* @author The OpenPilot Team, http://www.openpilot.org Copyright (C) 2012.
|
|
|
|
* @brief OSD gen module, handles OSD draw. Parts from CL-OSD and SUPEROSD projects
|
|
|
|
* @see The GNU Public License (GPL) Version 3
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
2013-04-07 09:49:13 +03:00
|
|
|
/*
|
2013-04-20 18:07:19 +03:00
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
2013-04-07 09:49:13 +03:00
|
|
|
*
|
2013-04-20 18:07:19 +03:00
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
|
|
|
|
* or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
2013-04-07 09:49:13 +03:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef OSDGEN_H_
|
|
|
|
#define OSDGEN_H_
|
|
|
|
|
|
|
|
#include "openpilot.h"
|
|
|
|
#include "pios.h"
|
|
|
|
|
|
|
|
int32_t osdgenInitialize(void);
|
|
|
|
|
|
|
|
// Size of an array (num items.)
|
|
|
|
#define SIZEOF_ARRAY(x) (sizeof(x) / sizeof((x)[0]))
|
|
|
|
|
2013-05-19 17:37:30 +03:00
|
|
|
#define HUD_VSCALE_FLAG_CLEAR 1
|
|
|
|
#define HUD_VSCALE_FLAG_NO_NEGATIVE 2
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Macros for computing addresses and bit positions.
|
|
|
|
// NOTE: /16 in y is because we are addressing by word not byte.
|
2013-05-19 17:37:30 +03:00
|
|
|
#define CALC_BUFF_ADDR(x, y) (((x) / 8) + ((y) * (GRAPHICS_WIDTH_REAL / 8)))
|
|
|
|
#define CALC_BIT_IN_WORD(x) ((x) & 7)
|
2013-04-07 09:49:13 +03:00
|
|
|
#define DEBUG_DELAY
|
|
|
|
// Macro for writing a word with a mode (NAND = clear, OR = set, XOR = toggle)
|
|
|
|
// at a given position
|
|
|
|
#define WRITE_WORD_MODE(buff, addr, mask, mode) \
|
2013-05-19 17:37:30 +03:00
|
|
|
switch (mode) { \
|
|
|
|
case 0: buff[addr] &= ~mask; break; \
|
|
|
|
case 1: buff[addr] |= mask; break; \
|
|
|
|
case 2: buff[addr] ^= mask; break; }
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
#define WRITE_WORD_NAND(buff, addr, mask) { buff[addr] &= ~mask; DEBUG_DELAY; }
|
|
|
|
#define WRITE_WORD_OR(buff, addr, mask) { buff[addr] |= mask; DEBUG_DELAY; }
|
|
|
|
#define WRITE_WORD_XOR(buff, addr, mask) { buff[addr] ^= mask; DEBUG_DELAY; }
|
|
|
|
|
|
|
|
// Horizontal line calculations.
|
|
|
|
// Edge cases.
|
2013-05-19 17:37:30 +03:00
|
|
|
#define COMPUTE_HLINE_EDGE_L_MASK(b) ((1 << (8 - (b))) - 1)
|
|
|
|
#define COMPUTE_HLINE_EDGE_R_MASK(b) (~((1 << (7 - (b))) - 1))
|
2013-04-07 09:49:13 +03:00
|
|
|
// This computes an island mask.
|
|
|
|
#define COMPUTE_HLINE_ISLAND_MASK(b0, b1) (COMPUTE_HLINE_EDGE_L_MASK(b0) ^ COMPUTE_HLINE_EDGE_L_MASK(b1));
|
|
|
|
|
|
|
|
// Macro for initializing stroke/fill modes. Add new modes here
|
|
|
|
// if necessary.
|
|
|
|
#define SETUP_STROKE_FILL(stroke, fill, mode) \
|
2013-05-19 17:37:30 +03:00
|
|
|
stroke = 0; fill = 0; \
|
|
|
|
if (mode == 0) { stroke = 0; fill = 1; } \
|
|
|
|
if (mode == 1) { stroke = 1; fill = 0; } \
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Line endcaps (for horizontal and vertical lines.)
|
2013-05-19 17:37:30 +03:00
|
|
|
#define ENDCAP_NONE 0
|
|
|
|
#define ENDCAP_ROUND 1
|
|
|
|
#define ENDCAP_FLAT 2
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
#define DRAW_ENDCAP_HLINE(e, x, y, s, f, l) \
|
2013-05-19 17:37:30 +03:00
|
|
|
if ((e) == ENDCAP_ROUND) /* single pixel endcap */ \
|
|
|
|
{ write_pixel_lm(x, y, f, l); } \
|
|
|
|
else if ((e) == ENDCAP_FLAT) /* flat endcap: FIXME, quicker to draw a vertical line(?) */ \
|
|
|
|
{ write_pixel_lm(x, y - 1, s, l); write_pixel_lm(x, y, s, l); write_pixel_lm(x, y + 1, s, l); }
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
#define DRAW_ENDCAP_VLINE(e, x, y, s, f, l) \
|
2013-05-19 17:37:30 +03:00
|
|
|
if ((e) == ENDCAP_ROUND) /* single pixel endcap */ \
|
|
|
|
{ write_pixel_lm(x, y, f, l); } \
|
|
|
|
else if ((e) == ENDCAP_FLAT) /* flat endcap: FIXME, quicker to draw a horizontal line(?) */ \
|
|
|
|
{ write_pixel_lm(x - 1, y, s, l); write_pixel_lm(x, y, s, l); write_pixel_lm(x + 1, y, s, l); }
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Macros for writing pixels in a midpoint circle algorithm.
|
|
|
|
#define CIRCLE_PLOT_8(buff, cx, cy, x, y, mode) \
|
2013-05-19 17:37:30 +03:00
|
|
|
CIRCLE_PLOT_4(buff, cx, cy, x, y, mode); \
|
|
|
|
if ((x) != (y)) { CIRCLE_PLOT_4(buff, cx, cy, y, x, mode); }
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
#define CIRCLE_PLOT_4(buff, cx, cy, x, y, mode) \
|
2013-05-19 17:37:30 +03:00
|
|
|
write_pixel(buff, (cx) + (x), (cy) + (y), mode); \
|
|
|
|
write_pixel(buff, (cx) - (x), (cy) + (y), mode); \
|
|
|
|
write_pixel(buff, (cx) + (x), (cy) - (y), mode); \
|
|
|
|
write_pixel(buff, (cx) - (x), (cy) - (y), mode);
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Font flags.
|
2013-05-19 17:37:30 +03:00
|
|
|
#define FONT_BOLD 1 // bold text (no outline)
|
|
|
|
#define FONT_INVERT 2 // invert: border white, inside black
|
2013-04-07 09:49:13 +03:00
|
|
|
// Text alignments.
|
2013-05-19 17:37:30 +03:00
|
|
|
#define TEXT_VA_TOP 0
|
|
|
|
#define TEXT_VA_MIDDLE 1
|
|
|
|
#define TEXT_VA_BOTTOM 2
|
|
|
|
#define TEXT_HA_LEFT 0
|
|
|
|
#define TEXT_HA_CENTER 1
|
|
|
|
#define TEXT_HA_RIGHT 2
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Text dimension structures.
|
2013-05-19 17:37:30 +03:00
|
|
|
struct FontDimensions {
|
2013-04-20 18:07:19 +03:00
|
|
|
int width, height;
|
2013-04-07 09:49:13 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
// Max/Min macros.
|
2013-05-19 17:37:30 +03:00
|
|
|
#define MAX(a, b) ((a) > (b) ? (a) : (b))
|
|
|
|
#define MIN(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
#define MAX3(a, b, c) MAX(a, MAX(b, c))
|
|
|
|
#define MIN3(a, b, c) MIN(a, MIN(b, c))
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Apply DeadBand
|
2013-05-19 17:37:30 +03:00
|
|
|
#define APPLY_DEADBAND(x, y) { x = (x) + GRAPHICS_HDEADBAND; y = (y) + GRAPHICS_VDEADBAND; }
|
|
|
|
#define APPLY_VDEADBAND(y) ((y) + GRAPHICS_VDEADBAND)
|
|
|
|
#define APPLY_HDEADBAND(x) ((x) + GRAPHICS_HDEADBAND)
|
2013-04-07 09:49:13 +03:00
|
|
|
|
2013-05-05 16:32:24 +09:30
|
|
|
// Check if coordinates are valid. If not, return. Assumes unsigned coordinate
|
2013-05-19 17:37:30 +03:00
|
|
|
#define CHECK_COORDS(x, y) if (x >= GRAPHICS_WIDTH_REAL || y >= GRAPHICS_HEIGHT_REAL) { return; }
|
|
|
|
#define CHECK_COORD_X(x) if (x >= GRAPHICS_WIDTH_REAL) { return; }
|
|
|
|
#define CHECK_COORD_Y(y) if (y >= GRAPHICS_HEIGHT_REAL) { return; }
|
2013-05-05 16:32:24 +09:30
|
|
|
|
|
|
|
// Clip coordinates out of range - assumes unsigned coordinate
|
2013-05-19 17:37:30 +03:00
|
|
|
#define CLIP_COORD_X(x) { x = MIN(x, GRAPHICS_WIDTH_REAL); }
|
|
|
|
#define CLIP_COORD_Y(y) { y = MIN(y, GRAPHICS_HEIGHT_REAL); }
|
|
|
|
#define CLIP_COORDS(x, y) { CLIP_COORD_X(x); CLIP_COORD_Y(y); }
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Macro to swap two variables using XOR swap.
|
2013-05-19 17:37:30 +03:00
|
|
|
#define SWAP(a, b) { a ^= b; b ^= a; a ^= b; }
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Line triggering
|
2013-05-19 17:37:30 +03:00
|
|
|
#define LAST_LINE 312 // 625/2 //PAL
|
|
|
|
// #define LAST_LINE 525/2 //NTSC
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
// Global vars
|
|
|
|
|
2013-05-19 17:37:30 +03:00
|
|
|
#define DELAY_1_NOP() asm ("nop\r\n")
|
|
|
|
#define DELAY_2_NOP() asm ("nop\r\nnop\r\n")
|
|
|
|
#define DELAY_3_NOP() asm ("nop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_4_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_5_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_6_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_7_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_8_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_9_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\n")
|
|
|
|
#define DELAY_10_NOP() asm ("nop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\nnop\r\n")
|
2013-04-07 09:49:13 +03:00
|
|
|
|
|
|
|
uint8_t getCharData(uint16_t charPos);
|
|
|
|
void introText();
|
|
|
|
|
|
|
|
void clearGraphics();
|
|
|
|
uint8_t validPos(uint16_t x, uint16_t y);
|
|
|
|
void setPixel(uint16_t x, uint16_t y, uint8_t state);
|
|
|
|
void drawCircle(uint16_t x0, uint16_t y0, uint16_t radius);
|
2013-05-19 17:37:30 +03:00
|
|
|
void swap(uint16_t *a, uint16_t *b);
|
2013-04-07 09:49:13 +03:00
|
|
|
void drawLine(uint16_t x0, uint16_t y0, uint16_t x1, uint16_t y1);
|
|
|
|
void drawBox(uint16_t x1, uint16_t y1, uint16_t x2, uint16_t y2);
|
|
|
|
void drawArrow(uint16_t x, uint16_t y, uint16_t angle, uint16_t size);
|
|
|
|
void drawAttitude(uint16_t x, uint16_t y, int16_t pitch, int16_t roll, uint16_t size);
|
|
|
|
void introGraphics();
|
|
|
|
void updateGraphics();
|
|
|
|
void drawGraphicsLine();
|
|
|
|
|
|
|
|
void write_char16(char ch, unsigned int x, unsigned int y, int font);
|
|
|
|
void write_pixel(uint8_t *buff, unsigned int x, unsigned int y, int mode);
|
|
|
|
void write_pixel_lm(unsigned int x, unsigned int y, int mmode, int lmode);
|
|
|
|
void write_hline(uint8_t *buff, unsigned int x0, unsigned int x1, unsigned int y, int mode);
|
|
|
|
void write_hline_lm(unsigned int x0, unsigned int x1, unsigned int y, int lmode, int mmode);
|
|
|
|
void write_hline_outlined(unsigned int x0, unsigned int x1, unsigned int y, int endcap0, int endcap1, int mode, int mmode);
|
|
|
|
void write_vline(uint8_t *buff, unsigned int x, unsigned int y0, unsigned int y1, int mode);
|
|
|
|
void write_vline_lm(unsigned int x, unsigned int y0, unsigned int y1, int lmode, int mmode);
|
|
|
|
void write_vline_outlined(unsigned int x, unsigned int y0, unsigned int y1, int endcap0, int endcap1, int mode, int mmode);
|
|
|
|
void write_filled_rectangle(uint8_t *buff, unsigned int x, unsigned int y, unsigned int width, unsigned int height, int mode);
|
|
|
|
void write_filled_rectangle_lm(unsigned int x, unsigned int y, unsigned int width, unsigned int height, int lmode, int mmode);
|
|
|
|
void write_rectangle_outlined(unsigned int x, unsigned int y, int width, int height, int mode, int mmode);
|
|
|
|
void write_circle(uint8_t *buff, unsigned int cx, unsigned int cy, unsigned int r, unsigned int dashp, int mode);
|
|
|
|
void write_circle_outlined(unsigned int cx, unsigned int cy, unsigned int r, unsigned int dashp, int bmode, int mode, int mmode);
|
|
|
|
void write_circle_filled(uint8_t *buff, unsigned int cx, unsigned int cy, unsigned int r, int mode);
|
|
|
|
void write_line(uint8_t *buff, unsigned int x0, unsigned int y0, unsigned int x1, unsigned int y1, int mode);
|
|
|
|
void write_line_lm(unsigned int x0, unsigned int y0, unsigned int x1, unsigned int y1, int mmode, int lmode);
|
|
|
|
void write_line_outlined(unsigned int x0, unsigned int y0, unsigned int x1, unsigned int y1, int endcap0, int endcap1, int mode, int mmode);
|
|
|
|
void write_word_misaligned(uint8_t *buff, uint16_t word, unsigned int addr, unsigned int xoff, int mode);
|
|
|
|
void write_word_misaligned_NAND(uint8_t *buff, uint16_t word, unsigned int addr, unsigned int xoff);
|
|
|
|
void write_word_misaligned_OR(uint8_t *buff, uint16_t word, unsigned int addr, unsigned int xoff);
|
|
|
|
void write_word_misaligned_lm(uint16_t wordl, uint16_t wordm, unsigned int addr, unsigned int xoff, int lmode, int mmode);
|
2013-05-19 17:37:30 +03:00
|
|
|
// int fetch_font_info(char ch, int font, struct FontEntry *font_info, char *lookup);
|
2013-04-07 09:49:13 +03:00
|
|
|
void write_char(char ch, unsigned int x, unsigned int y, int flags, int font);
|
2013-05-19 17:37:30 +03:00
|
|
|
// void calc_text_dimensions(char *str, struct FontEntry font, int xs, int ys, struct FontDimensions *dim);
|
2013-04-07 09:49:13 +03:00
|
|
|
void write_string(char *str, unsigned int x, unsigned int y, unsigned int xs, unsigned int ys, int va, int ha, int flags, int font);
|
|
|
|
void write_string_formatted(char *str, unsigned int x, unsigned int y, unsigned int xs, unsigned int ys, int va, int ha, int flags);
|
|
|
|
|
|
|
|
void updateOnceEveryFrame();
|
|
|
|
|
|
|
|
#endif /* OSDGEN_H_ */
|