2011-12-24 11:13:33 +04:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
2012-06-29 14:42:59 +04:00
|
|
|
/* vim: set sw=2 ts=8 et ft=cpp : */
|
2012-07-20 13:10:44 +04:00
|
|
|
/* Copyright 2012 Mozilla Foundation and Mozilla contributors
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2011-12-24 11:13:33 +04:00
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <linux/fb.h>
|
|
|
|
#include <linux/kd.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "android/log.h"
|
|
|
|
|
|
|
|
#include "Framebuffer.h"
|
2011-12-31 06:28:40 +04:00
|
|
|
#include "gfxContext.h"
|
2011-12-24 11:13:33 +04:00
|
|
|
#include "gfxImageSurface.h"
|
2011-12-31 06:28:40 +04:00
|
|
|
#include "gfxUtils.h"
|
2011-12-24 11:13:33 +04:00
|
|
|
#include "mozilla/FileUtils.h"
|
|
|
|
#include "nsTArray.h"
|
|
|
|
|
|
|
|
#define LOG(args...) __android_log_print(ANDROID_LOG_INFO, "Gonk" , ## args)
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
|
|
|
|
namespace Framebuffer {
|
|
|
|
|
|
|
|
static int sFd = -1;
|
|
|
|
static size_t sMappedSize;
|
|
|
|
static struct fb_var_screeninfo sVi;
|
|
|
|
static size_t sActiveBuffer;
|
|
|
|
typedef vector<nsRefPtr<gfxImageSurface> > BufferVector;
|
|
|
|
BufferVector* sBuffers;
|
2012-07-30 18:20:58 +04:00
|
|
|
static gfxIntSize *sScreenSize = nullptr;
|
2011-12-24 11:13:33 +04:00
|
|
|
|
|
|
|
BufferVector& Buffers() { return *sBuffers; }
|
|
|
|
|
|
|
|
bool
|
|
|
|
SetGraphicsMode()
|
|
|
|
{
|
|
|
|
ScopedClose fd(open("/dev/tty0", O_RDWR | O_SYNC));
|
2012-04-12 14:21:24 +04:00
|
|
|
if (0 > fd.get()) {
|
2011-12-24 11:13:33 +04:00
|
|
|
// This is non-fatal; post-Cupcake kernels don't have tty0.
|
|
|
|
LOG("No /dev/tty0?");
|
2012-04-12 14:21:24 +04:00
|
|
|
} else if (ioctl(fd.get(), KDSETMODE, (void*) KD_GRAPHICS)) {
|
2011-12-24 11:13:33 +04:00
|
|
|
LOG("Error setting graphics mode on /dev/tty0");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2012-06-26 18:38:50 +04:00
|
|
|
Open()
|
2011-12-24 11:13:33 +04:00
|
|
|
{
|
|
|
|
if (0 <= sFd)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (!SetGraphicsMode())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ScopedClose fd(open("/dev/graphics/fb0", O_RDWR));
|
2012-04-12 14:21:24 +04:00
|
|
|
if (0 > fd.get()) {
|
2011-12-24 11:13:33 +04:00
|
|
|
LOG("Error opening framebuffer device");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct fb_fix_screeninfo fi;
|
2012-04-12 14:21:24 +04:00
|
|
|
if (0 > ioctl(fd.get(), FBIOGET_FSCREENINFO, &fi)) {
|
2011-12-24 11:13:33 +04:00
|
|
|
LOG("Error getting fixed screeninfo");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-04-12 14:21:24 +04:00
|
|
|
if (0 > ioctl(fd.get(), FBIOGET_VSCREENINFO, &sVi)) {
|
2011-12-24 11:13:33 +04:00
|
|
|
LOG("Error getting variable screeninfo");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
sMappedSize = fi.smem_len;
|
|
|
|
void* mem = mmap(0, sMappedSize, PROT_READ | PROT_WRITE, MAP_SHARED,
|
2012-04-12 14:21:24 +04:00
|
|
|
fd.rwget(), 0);
|
2011-12-24 11:13:33 +04:00
|
|
|
if (MAP_FAILED == mem) {
|
|
|
|
LOG("Error mmap'ing framebuffer");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-04-12 14:21:24 +04:00
|
|
|
sFd = fd.get();
|
|
|
|
fd.forget();
|
2011-12-24 11:13:33 +04:00
|
|
|
|
|
|
|
// The android porting doc requires a /dev/graphics/fb0 device
|
|
|
|
// that's double buffered with r5g6b5 format. Hence the
|
|
|
|
// hard-coded numbers here.
|
|
|
|
gfxASurface::gfxImageFormat format = gfxASurface::ImageFormatRGB16_565;
|
|
|
|
int bytesPerPixel = gfxASurface::BytePerPixelFromFormat(format);
|
2012-06-26 18:38:50 +04:00
|
|
|
if (!sScreenSize) {
|
|
|
|
sScreenSize = new gfxIntSize(sVi.xres, sVi.yres);
|
|
|
|
}
|
2012-07-04 01:09:17 +04:00
|
|
|
long stride = fi.line_length;
|
2012-06-26 18:38:50 +04:00
|
|
|
size_t numFrameBytes = stride * sScreenSize->height;
|
2011-12-24 11:13:33 +04:00
|
|
|
|
|
|
|
sBuffers = new BufferVector(2);
|
|
|
|
unsigned char* data = static_cast<unsigned char*>(mem);
|
|
|
|
for (size_t i = 0; i < 2; ++i, data += numFrameBytes) {
|
|
|
|
memset(data, 0, numFrameBytes);
|
2012-06-26 18:38:50 +04:00
|
|
|
Buffers()[i] = new gfxImageSurface(data, *sScreenSize, stride, format);
|
2011-12-24 11:13:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// Clear the framebuffer to a known state.
|
2011-12-31 06:28:40 +04:00
|
|
|
Present(nsIntRect());
|
2011-12-24 11:13:33 +04:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-04-05 20:15:19 +04:00
|
|
|
bool
|
|
|
|
GetSize(nsIntSize *aScreenSize) {
|
2012-06-26 18:38:50 +04:00
|
|
|
// If the framebuffer has been opened, we should always have the size.
|
|
|
|
if (0 <= sFd || sScreenSize) {
|
|
|
|
*aScreenSize = *sScreenSize;
|
2012-04-05 20:15:19 +04:00
|
|
|
return true;
|
2012-06-26 18:38:50 +04:00
|
|
|
}
|
2012-04-05 20:15:19 +04:00
|
|
|
|
|
|
|
ScopedClose fd(open("/dev/graphics/fb0", O_RDWR));
|
2012-04-12 14:21:24 +04:00
|
|
|
if (0 > fd.get()) {
|
2012-04-05 20:15:19 +04:00
|
|
|
LOG("Error opening framebuffer device");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-04-12 14:21:24 +04:00
|
|
|
if (0 > ioctl(fd.get(), FBIOGET_VSCREENINFO, &sVi)) {
|
2012-04-05 20:15:19 +04:00
|
|
|
LOG("Error getting variable screeninfo");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-06-26 18:38:50 +04:00
|
|
|
sScreenSize = new gfxIntSize(sVi.xres, sVi.yres);
|
|
|
|
*aScreenSize = *sScreenSize;
|
2012-04-05 20:15:19 +04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2011-12-24 11:13:33 +04:00
|
|
|
void
|
|
|
|
Close()
|
|
|
|
{
|
|
|
|
if (0 > sFd)
|
|
|
|
return;
|
|
|
|
|
|
|
|
munmap(Buffers()[0]->Data(), sMappedSize);
|
|
|
|
delete sBuffers;
|
|
|
|
sBuffers = NULL;
|
2012-06-26 18:38:50 +04:00
|
|
|
delete sScreenSize;
|
|
|
|
sScreenSize = NULL;
|
2011-12-24 11:13:33 +04:00
|
|
|
|
|
|
|
close(sFd);
|
|
|
|
sFd = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
gfxASurface*
|
|
|
|
BackBuffer()
|
|
|
|
{
|
|
|
|
return Buffers()[!sActiveBuffer];
|
|
|
|
}
|
|
|
|
|
2011-12-31 06:28:40 +04:00
|
|
|
static gfxASurface*
|
|
|
|
FrontBuffer()
|
|
|
|
{
|
|
|
|
return Buffers()[sActiveBuffer];
|
|
|
|
}
|
|
|
|
|
2011-12-24 11:13:33 +04:00
|
|
|
void
|
2011-12-31 06:28:40 +04:00
|
|
|
Present(const nsIntRegion& aUpdated)
|
2011-12-24 11:13:33 +04:00
|
|
|
{
|
|
|
|
sActiveBuffer = !sActiveBuffer;
|
|
|
|
|
|
|
|
sVi.yres_virtual = sVi.yres * 2;
|
|
|
|
sVi.yoffset = sActiveBuffer * sVi.yres;
|
|
|
|
sVi.bits_per_pixel = 16;
|
|
|
|
if (ioctl(sFd, FBIOPUT_VSCREENINFO, &sVi) < 0) {
|
|
|
|
LOG("Error presenting front buffer");
|
|
|
|
}
|
2011-12-31 06:28:40 +04:00
|
|
|
|
|
|
|
nsRefPtr<gfxContext> ctx = new gfxContext(BackBuffer());
|
|
|
|
gfxUtils::PathFromRegion(ctx, aUpdated);
|
|
|
|
ctx->Clip();
|
|
|
|
ctx->SetSource(FrontBuffer());
|
|
|
|
ctx->SetOperator(gfxContext::OPERATOR_SOURCE);
|
|
|
|
ctx->Paint(1.0);
|
2011-12-24 11:13:33 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace Framebuffer
|
|
|
|
|
|
|
|
} // namespace mozilla
|