genode/demo/src/server/liquid_framebuffer/main.cc

339 lines
8.6 KiB
C++

/*
* \brief Nitpicker-based virtual framebuffer
* \author Norman Feske
* \date 2006-09-21
*/
/*
* Copyright (C) 2006-2013 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU General Public License version 2.
*/
#include <base/rpc_server.h>
#include <base/signal.h>
#include <cap_session/connection.h>
#include <os/config.h>
#include "framebuffer_window.h"
#include "canvas_rgb565.h"
#include "user_state.h"
#include "services.h"
using namespace Genode;
/**
* Runtime configuration
*/
namespace Config
{
int iconbar_detail = 1;
int background_detail = 1;
int mouse_cursor = 1;
int browser_attr = 0;
}
void Launcher::launch() { }
extern int native_startup(int, char **);
class Background_animator : public Tick
{
private:
Framebuffer_window<Pixel_rgb565> *_fb_win;
int _bg_offset;
public:
/**
* Constructor
*/
Background_animator(Framebuffer_window<Pixel_rgb565> *fb_win):
_fb_win(fb_win), _bg_offset(0) {
schedule(20); }
/**
* Tick interface
*/
int on_tick()
{
_fb_win->bg_offset(_bg_offset);
_bg_offset += 2;
_fb_win->refresh();
/* schedule next tick */
return 1;
}
};
/**
* Animated background
*/
static bool config_animate = true;
static bool config_alpha = true;
/**
* Size and position of virtual frame buffer
*/
static long config_fb_width = 500;
static long config_fb_height = 400;
static long config_fb_x = 400;
static long config_fb_y = 260;
/**
* Window title
*/
static const char *config_title = "Liquid Framebuffer";
/**
* Resize handle
*/
static bool config_resize_handle = false;
/**
* Window decoration
*/
static bool config_decoration = true;
/**
* Parse configuration
*/
static void read_config()
{
using namespace Genode;
Xml_node config_node = config()->xml_node();
try {
char buf[16];
config_node.attribute("animate").value(buf, sizeof(buf));
if (!strcmp("off", buf)) config_animate = false;
else if (!strcmp("on", buf)) config_animate = true;
else
Genode::printf("Warning: invalid value for animate declaration,\n"
" valid values are 'on', 'off.\n'");
} catch (Xml_node::Nonexistent_attribute) { }
config_alpha = config_animate;
try { config_node.attribute("xpos").value(&config_fb_x); }
catch (Xml_node::Nonexistent_attribute) { }
try { config_node.attribute("ypos").value(&config_fb_y); }
catch (Xml_node::Nonexistent_attribute) { }
try { config_node.attribute("width").value(&config_fb_width); }
catch (Xml_node::Nonexistent_attribute) { }
try { config_node.attribute("height").value(&config_fb_height); }
catch (Xml_node::Nonexistent_attribute) { }
try {
static char buf[64];
config_node.attribute("title").value(buf, sizeof(buf));
config_title = buf;
} catch (Xml_node::Nonexistent_attribute) { }
try {
char buf[16];
config_node.attribute("decoration").value(buf, sizeof(buf));
if (!strcmp("off", buf)) config_decoration = false;
else if (!strcmp("on", buf)) config_decoration = true;
else
Genode::printf("Warning: invalid value for decoration declaration,\n"
" valid values are 'on', 'off.\n'");
} catch (Xml_node::Nonexistent_attribute) { }
try {
char buf[16];
config_node.attribute("resize_handle").value(buf, sizeof(buf));
if (!strcmp("off", buf)) config_resize_handle = false;
else if (!strcmp("on", buf)) config_resize_handle = true;
else
Genode::printf("Warning: invalid value for resize_handle declaration,\n"
" valid values are 'on', 'off.\n'");
} catch (Xml_node::Nonexistent_attribute) { }
}
/*******************
** Input handler **
*******************/
struct Input_handler
{
GENODE_RPC(Rpc_handle_input, void, handle, Event&);
GENODE_RPC_INTERFACE(Rpc_handle_input);
};
class Input_handler_component : public Genode::Rpc_object<Input_handler,
Input_handler_component>
{
private:
Platform &_pf;
User_state &_user_state;
Framebuffer_window<Pixel_rgb565> &_fb_win;
Redraw_manager &_redraw;
Signal_receiver &_sig_rec;
unsigned long _curr_time, _old_time;
public:
Input_handler_component(Platform &pf,
User_state &user_state,
Framebuffer_window<Pixel_rgb565> &fb_win,
Redraw_manager &redraw,
Signal_receiver &sig_rec)
: _pf(pf),
_user_state(user_state),
_fb_win(fb_win),
_redraw(redraw),
_sig_rec(sig_rec)
{
_curr_time = _old_time = _pf.timer_ticks();
}
void handle(Event &ev)
{
if (ev.type != Event::WHEEL) {
ev.mx -= _user_state.vx();
ev.my -= _user_state.vy();
}
/* direct all keyboard events to the window content */
if ((ev.type == Event::PRESS || ev.type == Event::RELEASE)
&& (ev.code != Event::BTN_LEFT))
window_content()->handle_event(ev);
else
_user_state.handle_event(ev);
if (ev.type == Event::REFRESH)
_pf.scr_update(0, 0, _pf.scr_w(), _pf.scr_h());
if (ev.type == Event::TIMER) {
Tick::handle(_pf.timer_ticks());
/* check for configuration changes */
if (_sig_rec.pending()) {
_sig_rec.wait_for_signal();
config()->reload();
/* keep the current values by default */
config_fb_x = _fb_win.view_x();
config_fb_y = _fb_win.view_y();
config_fb_width = _fb_win.view_w();
config_fb_height = _fb_win.view_h();
try { read_config(); } catch (...) { }
_fb_win.name(config_title);
_fb_win.config_alpha(config_alpha);
_fb_win.config_resize_handle(config_resize_handle);
_fb_win.config_decoration(config_decoration);
/* must get called after 'config_decoration()' */
_fb_win.content_geometry(config_fb_x, config_fb_y,
config_fb_width, config_fb_height);
_user_state.update_view_offset();
}
}
/* perform periodic redraw */
_curr_time = _pf.timer_ticks();
if (!_pf.event_pending() && ((_curr_time - _old_time > 20) || (_curr_time < _old_time))) {
_old_time = _curr_time;
_redraw.process();
}
}
};
/**
* Main program
*/
int main(int argc, char **argv)
{
if (native_startup(argc, argv)) return -1;
try { read_config(); } catch (...) { }
/*
* Register signal handler for config changes
*/
static Signal_receiver sig_rec;
static Signal_context sig_ctx;
try { config()->sigh(sig_rec.manage(&sig_ctx)); } catch (...) { }
/* heuristic for allocating the double-buffer backing store */
enum { WINBORDER_WIDTH = 10, WINBORDER_HEIGHT = 40 };
/* init platform */
static Platform pf(config_fb_x, config_fb_y,
config_fb_width + WINBORDER_WIDTH,
config_fb_height + WINBORDER_HEIGHT,
config_fb_width + WINBORDER_WIDTH,
config_fb_height + WINBORDER_HEIGHT);
/* initialize our window content */
init_window_content(config_fb_width, config_fb_height, config_alpha);
/* init canvas */
static Chunky_canvas<Pixel_rgb565> canvas;
canvas.init(static_cast<Pixel_rgb565 *>(pf.buf_adr()),
pf.scr_w()*pf.scr_h());
canvas.set_size(pf.scr_w(), pf.scr_h());
canvas.clip(0, 0, pf.scr_w(), pf.scr_h());
/* init redraw manager */
static Redraw_manager redraw(&canvas, &pf, pf.vw(), pf.vh());
/* create instance of browser window */
static Framebuffer_window<Pixel_rgb565>
fb_win(&pf, &redraw, window_content(), config_title, config_alpha,
config_resize_handle, config_decoration);
if (config_animate) {
static Background_animator fb_win_bg_anim(&fb_win);
}
/* create user state manager */
static User_state user_state(&fb_win, &fb_win, pf.vx(), pf.vy());
/* assign framebuffer window as root element to redraw manager */
redraw.root(&fb_win);
fb_win.parent(&user_state);
fb_win.content_geometry(config_fb_x, config_fb_y,
config_fb_width, config_fb_height);
/* initialize server entry point */
enum { STACK_SIZE = 2*1024*sizeof(addr_t) };
static Cap_connection cap;
static Rpc_entrypoint ep(&cap, STACK_SIZE, "liquid_fb_ep");
/* initialize public services */
init_services(ep);
/* create local input handler service */
static Input_handler_component input_handler(pf, user_state, fb_win,
redraw, sig_rec);
Capability<Input_handler> input_handler_cap = ep.manage(&input_handler);
/* enter main loop */
Event ev;
do {
pf.get_event(&ev);
input_handler_cap.call<Input_handler::Rpc_handle_input>(ev);
} while (ev.type != Event::QUIT);
return 0;
}