upiwin/src/fbinit.c

111 lines
2.6 KiB
C
Raw Normal View History

2019-12-01 18:53:54 -07:00
#include <stddef.h>
#include <string.h>
2019-12-01 18:53:54 -07:00
#include <fcntl.h>
2019-12-01 19:20:04 -07:00
#include <unistd.h>
2019-12-01 18:53:54 -07:00
#include <errno.h>
#include <sys/ioctl.h>
#include <sys/mman.h>
2019-12-01 18:53:54 -07:00
#include <linux/fb.h>
#include "log.h"
#include "fbinit.h"
#include "scode.h"
2019-12-01 18:53:54 -07:00
/* references to splash screen data in splash.o/splash.bin */
extern uint8_t _binary_splash_bin_start[];
extern uint8_t _binary_splash_bin_end;
extern uint8_t _binary_splash_bin_size;
2019-12-01 18:53:54 -07:00
static int fb_fd = -1;
static FBINFO local_info;
PCFBINFO Fb_Info = &local_info;
2019-12-01 18:53:54 -07:00
UINT16 *Fb_Ptr = NULL;
inline static UINT16 makemask(unsigned offset, unsigned length)
2019-12-01 18:53:54 -07:00
{
return (UINT16)(((1 << length) - 1) << offset);
2019-12-01 18:53:54 -07:00
}
HRESULT Fb_setup(void)
2019-12-01 18:53:54 -07:00
{
HRESULT hr = S_OK;
2019-12-01 18:53:54 -07:00
struct fb_fix_screeninfo fixed;
struct fb_var_screeninfo var;
fb_fd = open("/dev/fb1", O_RDWR);
if (fb_fd == -1)
{
hr = ERRNO_AS_SCODE;
Log(LFATAL, "Unable to open framebuffer (%08X)", hr);
return hr;
2019-12-01 18:53:54 -07:00
}
if (ioctl(fb_fd, FBIOGET_FSCREENINFO, &fixed))
{
hr = ERRNO_AS_SCODE;
Log(LFATAL, "Could not get fixed screen info (%08X)", hr);
return hr;
2019-12-01 18:53:54 -07:00
}
local_info.linebytes = fixed.line_length;
local_info.screenbytes = fixed.smem_len;
2019-12-01 18:53:54 -07:00
if (ioctl(fb_fd, FBIOGET_VSCREENINFO, &var))
{
hr = ERRNO_AS_SCODE;
Log(LFATAL, "Could not get variable screen info (%08X)", hr);
return hr;
2019-12-01 18:53:54 -07:00
}
local_info.width = var.xres;
local_info.height = var.yres;
local_info.virtual_width = var.xres_virtual;
local_info.virtual_height = var.yres_virtual;
local_info.bpp = var.bits_per_pixel;
local_info.red_offset = var.red.offset;
local_info.red_length = var.red.length;
local_info.red_mask = makemask(var.red.offset, var.red.length);
local_info.green_offset = var.green.offset;
local_info.green_length = var.green.length;
local_info.green_mask = makemask(var.green.offset, var.green.length);
local_info.blue_offset = var.blue.offset;
local_info.blue_length = var.blue.length;
local_info.blue_mask = makemask(var.blue.offset, var.blue.length);
Fb_Ptr = (UINT16 *)mmap(0, fixed.smem_len, PROT_READ|PROT_WRITE, MAP_SHARED, fb_fd, 0);
if ((int)Fb_Ptr == -1)
{
hr = ERRNO_AS_SCODE;
Log(LFATAL, "Unable to memmap framebuffer (%08X)", hr);
Fb_Ptr = NULL;
close(fb_fd);
fb_fd = -1;
return hr;
}
2019-12-01 18:53:54 -07:00
/* display the splash screen */
memcpy(Fb_ptr, _binary_splash_bin_start, (size_t)(&_binary_splash_bin_size));
/* additional setup here */
2019-12-01 18:53:54 -07:00
return hr;
2019-12-01 18:53:54 -07:00
}
void Fb_cleanup(void)
{
/* additional cleanup here */
memset(Fb_Ptr, 0, local_info.screenbytes);
munmap((void *)Fb_Ptr, local_info.screenbytes);
Fb_Ptr = NULL;
2019-12-01 18:53:54 -07:00
close(fb_fd);
fb_fd = -1;
2019-12-01 18:53:54 -07:00
}
void Fb_clear(void)
{
memset(Fb_Ptr, 0, local_info.screenbytes);
}