You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
159 lines
4.0 KiB
159 lines
4.0 KiB
/* |
|
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. |
|
|
|
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, see <http://www.gnu.org/licenses/>. |
|
*/ |
|
#ifdef WITH_SITL_OSD |
|
|
|
#include "Display_SITL.h" |
|
|
|
#include <AP_HAL/AP_HAL.h> |
|
#include <AP_Notify/AP_Notify.h> |
|
|
|
#include <stdio.h> |
|
#include <unistd.h> |
|
|
|
// constructor |
|
Display_SITL::Display_SITL() |
|
{ |
|
} |
|
|
|
Display_SITL::~Display_SITL() |
|
{ |
|
} |
|
|
|
Display_SITL *Display_SITL::probe() |
|
{ |
|
Display_SITL *driver = new Display_SITL(); |
|
if (!driver || !driver->hw_init()) { |
|
delete driver; |
|
return nullptr; |
|
} |
|
return driver; |
|
} |
|
|
|
// main loop of graphics thread |
|
void Display_SITL::update_thread(void) |
|
{ |
|
{ |
|
WITH_SEMAPHORE(AP::notify().sf_window_mutex); |
|
w = new sf::RenderWindow(sf::VideoMode(COLUMNS*SCALE, ROWS*SCALE), "Display"); |
|
} |
|
if (!w) { |
|
AP_HAL::panic("Unable to create Display_SITL window"); |
|
} |
|
|
|
const sf::Color color_black = sf::Color(0,0,0); |
|
const sf::Color color_white = sf::Color(255,255,255); |
|
|
|
const sf::Uint8 pixels[ROWS*COLUMNS*4]{}; |
|
sf::Image image; |
|
image.create(COLUMNS, ROWS, pixels); |
|
|
|
while (true) { |
|
{ |
|
WITH_SEMAPHORE(AP::notify().sf_window_mutex); |
|
sf::Event event; |
|
while (w->pollEvent(event)) { |
|
if (event.type == sf::Event::Closed) { |
|
w->close(); |
|
} |
|
} |
|
if (!w->isOpen()) { |
|
break; |
|
} |
|
if (_need_hw_update) { |
|
_need_hw_update = false; |
|
|
|
uint8_t buffer2[ROWS*COLUMNS]; |
|
{ |
|
WITH_SEMAPHORE(mutex); |
|
memcpy(buffer2, _displaybuffer, sizeof(buffer2)); |
|
} |
|
w->clear(); |
|
|
|
for (uint16_t y=0; y<ROWS; y++) { |
|
for (uint16_t x=0; x<COLUMNS; x++) { |
|
if (buffer2[x+y/8*COLUMNS] & 1<<y%8) { |
|
image.setPixel(x, y, color_white); |
|
} else { |
|
image.setPixel(x, y, color_black); |
|
} |
|
} |
|
} |
|
|
|
sf::Texture texture; |
|
texture.loadFromImage(image); |
|
sf::Sprite sprite; |
|
sprite.setTexture(texture, true); |
|
sprite.setScale(SCALE, SCALE); |
|
w->draw(sprite); |
|
|
|
w->display(); |
|
} |
|
} |
|
usleep(10000); |
|
} |
|
} |
|
|
|
// trampoline for update thread |
|
void *Display_SITL::update_thread_start(void *obj) |
|
{ |
|
((Display_SITL *)obj)->update_thread(); |
|
return nullptr; |
|
} |
|
|
|
bool Display_SITL::hw_init() |
|
{ |
|
pthread_create(&thread, NULL, update_thread_start, this); |
|
_need_hw_update = true; |
|
|
|
return true; |
|
} |
|
|
|
void Display_SITL::hw_update() |
|
{ |
|
_need_hw_update = true; |
|
} |
|
|
|
void Display_SITL::set_pixel(uint16_t x, uint16_t y) |
|
{ |
|
// check x, y range |
|
if ((x >= COLUMNS) || (y >= ROWS)) { |
|
return; |
|
} |
|
// set pixel in buffer |
|
WITH_SEMAPHORE(mutex); |
|
_displaybuffer[x + (y / 8 * COLUMNS)] |= 1 << (y % 8); |
|
_need_hw_update = true; |
|
} |
|
|
|
void Display_SITL::clear_pixel(uint16_t x, uint16_t y) |
|
{ |
|
// check x, y range |
|
if ((x >= COLUMNS) || (y >= ROWS)) { |
|
return; |
|
} |
|
// clear pixel in buffer |
|
WITH_SEMAPHORE(mutex); |
|
_displaybuffer[x + (y / 8 * COLUMNS)] &= ~(1 << (y % 8)); |
|
_need_hw_update = true; |
|
} |
|
|
|
void Display_SITL::clear_screen() |
|
{ |
|
WITH_SEMAPHORE(mutex); |
|
memset(_displaybuffer, 0, sizeof(_displaybuffer)); |
|
_need_hw_update = true; |
|
} |
|
|
|
#endif // WITH_SITL_OSD
|
|
|