/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: sw=2 ts=8 et ft=cpp : */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (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.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is Mozilla Code. * * The Initial Developer of the Original Code is * The Mozilla Foundation * Portions created by the Initial Developer are Copyright (C) 2011 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Chris Jones * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include #include #include #include #include #include #include #include #include #include #include "android/log.h" #include "Framebuffer.h" #include "gfxContext.h" #include "gfxImageSurface.h" #include "gfxUtils.h" #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 > BufferVector; BufferVector* sBuffers; BufferVector& Buffers() { return *sBuffers; } bool SetGraphicsMode() { ScopedClose fd(open("/dev/tty0", O_RDWR | O_SYNC)); if (0 > fd.mFd) { // This is non-fatal; post-Cupcake kernels don't have tty0. LOG("No /dev/tty0?"); } else if (ioctl(fd.mFd, KDSETMODE, (void*) KD_GRAPHICS)) { LOG("Error setting graphics mode on /dev/tty0"); return false; } return true; } bool Open(nsIntSize* aScreenSize) { if (0 <= sFd) return true; if (!SetGraphicsMode()) return false; ScopedClose fd(open("/dev/graphics/fb0", O_RDWR)); if (0 > fd.mFd) { LOG("Error opening framebuffer device"); return false; } struct fb_fix_screeninfo fi; if (0 > ioctl(fd.mFd, FBIOGET_FSCREENINFO, &fi)) { LOG("Error getting fixed screeninfo"); return false; } if (0 > ioctl(fd.mFd, FBIOGET_VSCREENINFO, &sVi)) { LOG("Error getting variable screeninfo"); return false; } sMappedSize = fi.smem_len; void* mem = mmap(0, sMappedSize, PROT_READ | PROT_WRITE, MAP_SHARED, fd.mFd, 0); if (MAP_FAILED == mem) { LOG("Error mmap'ing framebuffer"); return false; } sFd = fd.mFd; fd.mFd = -1; // 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); gfxIntSize size(sVi.xres, sVi.yres); long stride = size.width * bytesPerPixel; size_t numFrameBytes = stride * size.height; sBuffers = new BufferVector(2); unsigned char* data = static_cast(mem); for (size_t i = 0; i < 2; ++i, data += numFrameBytes) { memset(data, 0, numFrameBytes); Buffers()[i] = new gfxImageSurface(data, size, stride, format); } // Clear the framebuffer to a known state. Present(nsIntRect()); *aScreenSize = size; return true; } void Close() { if (0 > sFd) return; munmap(Buffers()[0]->Data(), sMappedSize); delete sBuffers; sBuffers = NULL; close(sFd); sFd = -1; } gfxASurface* BackBuffer() { return Buffers()[!sActiveBuffer]; } static gfxASurface* FrontBuffer() { return Buffers()[sActiveBuffer]; } void Present(const nsIntRegion& aUpdated) { 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"); } nsRefPtr ctx = new gfxContext(BackBuffer()); gfxUtils::PathFromRegion(ctx, aUpdated); ctx->Clip(); ctx->SetSource(FrontBuffer()); ctx->SetOperator(gfxContext::OPERATOR_SOURCE); ctx->Paint(1.0); } } // namespace Framebuffer } // namespace mozilla