/****************************************************************************
**
** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies).
** All rights reserved.
** Contact: Nokia Corporation (qt-info@nokia.com)
**
** This file is part of the QtGui module of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:LGPL$
** No Commercial Usage
** This file contains pre-release code and may not be distributed.
** You may use this file in accordance with the terms and conditions
** contained in the Technology Preview License Agreement accompanying
** this package.
**
** GNU Lesser General Public License Usage
** Alternatively, this file may be used under the terms of the GNU Lesser
** General Public License version 2.1 as published by the Free Software
** Foundation and appearing in the file LICENSE.LGPL included in the
** packaging of this file. Please review the following information to
** ensure the GNU Lesser General Public License version 2.1 requirements
** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
**
** In addition, as a special exception, Nokia gives you certain additional
** rights. These rights are described in the Nokia Qt LGPL Exception
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
**
** If you have questions regarding the use of this file, please contact
** Nokia at qt-info@nokia.com.
**
**
**
**
**
**
**
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "qfontengine_p.h"
#include <private/qunicodetables_p.h>
#include <qwsdisplay_qws.h>
#include <qvarlengtharray.h>
#include <private/qpainter_p.h>
#include <private/qpaintengine_raster_p.h>
#include <private/qpdf_p.h>
#include "qtextengine_p.h"
#include "private/qcore_unix_p.h" // overrides QT_OPEN
#include <qdebug.h>
#ifndef QT_NO_QWS_QPF
#include "qfile.h"
#include "qdir.h"
#define QT_USE_MMAP
#include <stdlib.h>
#ifdef QT_USE_MMAP
// for mmap
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <errno.h>
# if defined(QT_LINUXBASE) && !defined(MAP_FILE)
// LSB 3.2 does not define MAP_FILE
# define MAP_FILE 0
# endif
#endif
#endif // QT_NO_QWS_QPF
QT_BEGIN_NAMESPACE
#ifndef QT_NO_QWS_QPF
QT_BEGIN_INCLUDE_NAMESPACE
#include "qplatformdefs.h"
QT_END_INCLUDE_NAMESPACE
static inline unsigned int getChar(const QChar *str, int &i, const int len)
{
unsigned int uc = str[i].unicode();
if (uc >= 0xd800 && uc < 0xdc00 && i < len-1) {
uint low = str[i+1].unicode();
if (low >= 0xdc00 && low < 0xe000) {
uc = (uc - 0xd800)*0x400 + (low - 0xdc00) + 0x10000;
++i;
}
}
return uc;
}
#define FM_SMOOTH 1
class Q_PACKED QPFGlyphMetrics {
public:
quint8 linestep;
quint8 width;
quint8 height;
quint8 flags;
qint8 bearingx; // Difference from pen position to glyph's left bbox
quint8 advance; // Difference between pen positions
qint8 bearingy; // Used for putting characters on baseline
qint8 reserved; // Do not use
// Flags:
// RendererOwnsData - the renderer is responsible for glyph data
// memory deletion otherwise QPFGlyphTree must
// delete [] the data when the glyph is deleted.
enum Flags { RendererOwnsData=0x01 };
};
class QPFGlyph {
public:
QPFGlyph() { metrics=0; data=0; }
QPFGlyph(QPFGlyphMetrics* m, uchar* d) :
metrics(m), data(d) { }
~QPFGlyph() {}
QPFGlyphMetrics* metrics;
uchar* data;
};
struct Q_PACKED QPFFontMetrics{
qint8 ascent,descent;
qint8 leftbearing,rightbearing;
quint8 maxwidth;
qint8 leading;
quint8 flags;
quint8 underlinepos;
quint8 underlinewidth;
quint8 reserved3;
};
class QPFGlyphTree {
public:
/* reads in a tree like this:
A-Z
/ \
0-9 a-z
etc.
*/
glyph_t min,max;
QPFGlyphTree* less;
QPFGlyphTree* more;
QPFGlyph* glyph;
public:
#ifdef QT_USE_MMAP
QPFGlyphTree(uchar*& data)
{
read(data);
}
#else
QPFGlyphTree(QIODevice& f)
{
read(f);
}
#endif
~QPFGlyphTree()
{
// NOTE: does not delete glyph[*].metrics or .data.
// the caller does this (only they know who owns
// the data). See clear().
delete less;
delete more;
delete [] glyph;
}
bool inFont(glyph_t g) const
{
if ( g < min ) {
if ( !less )
return false;
return less->inFont(g);
} else if ( g > max ) {
if ( !more )
return false;
return more->inFont(g);
}
return true;
}
QPFGlyph* get(glyph_t g)
{
if ( g < min ) {
if ( !less )
return 0;
return less->get(g);
} else if ( g > max ) {
if ( !more )
return 0;
return more->get(g);
}
return &glyph[g - min];
}
int totalChars() const
{
if ( !this ) return 0;
return max-min+1 + less->totalChars() + more->totalChars();
}
int weight() const
{
if ( !this ) return 0;
return 1 + less->weight() + more->weight();
}
void dump(int indent=0)
{
for (int i=0; i<indent; i++) printf(" ");
printf("%d..%d",min,max);
//if ( indent == 0 )
printf(" (total %d)",totalChars());
printf("\n");
if ( less ) less->dump(indent+1);
if ( more ) more->dump(indent+1);
}
private:
QPFGlyphTree()
{
}
#ifdef QT_USE_MMAP
void read(uchar*& data)
{
// All node data first
readNode(data);
// Then all non-video data
readMetrics(data);
// Then all video data
readData(data);
}
#else
void read(QIODevice& f)
{
// All node data first
readNode(f);
// Then all non-video data
readMetrics(f);
// Then all video data
readData(f);
}
#endif
#ifdef QT_USE_MMAP
void readNode(uchar*& data)
{
uchar rw = *data++;
uchar cl = *data++;
min = (rw << 8) | cl;
rw = *data++;
cl = *data++;
max = (rw << 8) | cl;
int flags = *data++;
if ( flags & 1 )
less = new QPFGlyphTree;
else
less = 0;
if ( flags & 2 )
more = new QPFGlyphTree;
else
more = 0;
int n = max-min+1;
glyph = new QPFGlyph[n];
if ( less )
less->readNode(data);
if ( more )
more->readNode(data);
}
#else
void readNode(QIODevice& f)
{
uchar rw = f.getch();
uchar cl = f.getch();
min = (rw << 8) | cl;
rw = f.getch();
cl = f.getch();
max = (rw << 8) | cl;
int flags = f.getch();
if ( flags & 1 )
less = new QPFGlyphTree;
else
less = 0;
if ( flags & 2 )
more = new QPFGlyphTree;
else
more = 0;
int n = max-min+1;
glyph = new QPFGlyph[n];
if ( less )
less->readNode(f);
if ( more )
more->readNode(f);
}
#endif
#ifdef QT_USE_MMAP
void readMetrics(uchar*& data)
{
int n = max-min+1;
for (int i=0; i<n; i++) {
glyph[i].metrics = (QPFGlyphMetrics*)data;
data += sizeof(QPFGlyphMetrics);
}
if ( less )
less->readMetrics(data);
if ( more )
more->readMetrics(data);
}
#else
void readMetrics(QIODevice& f)
{
int n = max-min+1;
for (int i=0; i<n; i++) {
glyph[i].metrics = new QPFGlyphMetrics;
f.readBlock((char*)glyph[i].metrics, sizeof(QPFGlyphMetrics));
}
if ( less )
less->readMetrics(f);
if ( more )
more->readMetrics(f);
}
#endif
#ifdef QT_USE_MMAP
void readData(uchar*& data)
{
int n = max-min+1;
for (int i=0; i<n; i++) {
QSize s( glyph[i].metrics->width, glyph[i].metrics->height );
//######### s = qt_screen->mapToDevice( s );
uint datasize = glyph[i].metrics->linestep * s.height();
glyph[i].data = data; data += datasize;
}
if ( less )
less->readData(data);
if ( more )
more->readData(data);
}
#else
void readData(QIODevice& f)
{
int n = max-min+1;
for (int i=0; i<n; i++) {
QSize s( glyph[i].metrics->width, glyph[i].metrics->height );
//############### s = qt_screen->mapToDevice( s );
uint datasize = glyph[i].metrics->linestep * s.height();
glyph[i].data = new uchar[datasize]; // ### deleted?
f.readBlock((char*)glyph[i].data, datasize);
}
if ( less )
less->readData(f);
if ( more )
more->readData(f);
}
#endif
};
class QFontEngineQPF1Data
{
public:
QPFFontMetrics fm;
QPFGlyphTree *tree;
void *mmapStart;
size_t mmapLength;
};
QFontEngineQPF1::QFontEngineQPF1(const QFontDef&, const QString &fn)
{
cache_cost = 1;
int f = QT_OPEN( QFile::encodeName(fn), O_RDONLY, 0);
Q_ASSERT(f>=0);
QT_STATBUF st;
if ( QT_FSTAT( f, &st ) )
qFatal("Failed to stat %s",QFile::encodeName(fn).data());
uchar* data = (uchar*)mmap( 0, // any address
st.st_size, // whole file
PROT_READ, // read-only memory
#if !defined(Q_OS_SOLARIS) && !defined(Q_OS_QNX4) && !defined(Q_OS_INTEGRITY) && !defined(Q_OS_VXWORKS)
MAP_FILE | MAP_PRIVATE, // swap-backed map from file
#else
MAP_PRIVATE,
#endif
f, 0 ); // from offset 0 of f
#if defined(Q_OS_QNX4) && !defined(MAP_FAILED)
#define MAP_FAILED ((void *)-1)
#endif
if ( !data || data == (uchar*)MAP_FAILED )
qFatal("Failed to mmap %s",QFile::encodeName(fn).data());
QT_CLOSE(f);
d = new QFontEngineQPF1Data;
d->mmapStart = data;
d->mmapLength = st.st_size;
memcpy(reinterpret_cast<char*>(&d->fm),data,sizeof(d->fm));
data += sizeof(d->fm);
d->tree = new QPFGlyphTree(data);
glyphFormat = (d->fm.flags & FM_SMOOTH) ? QFontEngineGlyphCache::Raster_A8
: QFontEngineGlyphCache::Raster_Mono;
#if 0
qDebug() << "font file" << fn
<< "ascent" << d->fm.ascent << "descent" << d->fm.descent
<< "leftbearing" << d->fm.leftbearing
<< "rightbearing" << d->fm.rightbearing
<< "maxwidth" << d->fm.maxwidth
<< "leading" << d->fm.leading
<< "flags" << d->fm.flags
<< "underlinepos" << d->fm.underlinepos
<< "underlinewidth" << d->fm.underlinewidth;
#endif
}
QFontEngineQPF1::~QFontEngineQPF1()
{
if (d->mmapStart)
munmap(d->mmapStart, d->mmapLength);
delete d->tree;
delete d;
}
bool QFontEngineQPF1::stringToCMap(const QChar *str, int len, QGlyphLayout *glyphs, int *nglyphs, QTextEngine::ShaperFlags flags) const
{
if(*nglyphs < len) {
*nglyphs = len;
return false;
}
*nglyphs = 0;
bool mirrored = flags & QTextEngine::RightToLeft;
for(int i = 0; i < len; i++) {
unsigned int uc = getChar(str, i, len);
if (mirrored)
uc = QChar::mirroredChar(uc);
glyphs->glyphs[*nglyphs] = uc < 0x10000 ? uc : 0;
++*nglyphs;
}
glyphs->numGlyphs = *nglyphs;
if (flags & QTextEngine::GlyphIndicesOnly)
return true;
recalcAdvances(glyphs, flags);
return true;
}
void QFontEngineQPF1::recalcAdvances(QGlyphLayout *glyphs, QTextEngine::ShaperFlags) const
{
for(int i = 0; i < glyphs->numGlyphs; i++) {
QPFGlyph *glyph = d->tree->get(glyphs->glyphs[i]);
glyphs->advances_x[i] = glyph ? glyph->metrics->advance : 0;
glyphs->advances_y[i] = 0;
if (!glyph)
glyphs->glyphs[i] = 0;
}
}
void QFontEngineQPF1::draw(QPaintEngine *p, qreal _x, qreal _y, const QTextItemInt &si)
{
QPaintEngineState *pState = p->state;
QRasterPaintEngine *paintEngine = static_cast<QRasterPaintEngine*>(p);
QTransform matrix = pState->transform();
matrix.translate(_x, _y);
QFixed x = QFixed::fromReal(matrix.dx());
QFixed y = QFixed::fromReal(matrix.dy());
QVarLengthArray<QFixedPoint> positions;
QVarLengthArray<glyph_t> glyphs;
getGlyphPositions(si.glyphs, matrix, si.flags, glyphs, positions);
if (glyphs.size() == 0)
return;
int depth = (d->fm.flags & FM_SMOOTH) ? 8 : 1;
for(int i = 0; i < glyphs.size(); i++) {
const QPFGlyph *glyph = d->tree->get(glyphs[i]);
if (!glyph)
continue;
int bpl = glyph->metrics->linestep;
if(glyph->data)
paintEngine->alphaPenBlt(glyph->data, bpl, depth,
qRound(positions[i].x) + glyph->metrics->bearingx,
qRound(positions[i].y) - glyph->metrics->bearingy,
glyph->metrics->width,glyph->metrics->height);
}
}
QImage QFontEngineQPF1::alphaMapForGlyph(glyph_t g)
{
const QPFGlyph *glyph = d->tree->get(g);
if (!glyph)
return QImage();
int mono = !(d->fm.flags & FM_SMOOTH);
const uchar *bits = glyph->data;//((const uchar *) glyph);
QImage image;
if (mono) {
image = QImage((glyph->metrics->width+7)&~7, glyph->metrics->height, QImage::Format_Mono);
image.setColor(0, qRgba(0, 0, 0, 0));
image.setColor(1, qRgba(0, 0, 0, 255));
} else {
image = QImage(glyph->metrics->width, glyph->metrics->height, QImage::Format_Indexed8);
for (int j=0; j<256; ++j)
image.setColor(j, qRgba(0, 0, 0, j));
}
for (int i=0; i<glyph->metrics->height; ++i) {
memcpy(image.scanLine(i), bits, glyph->metrics->linestep);
bits += glyph->metrics->linestep;
}
return image;
}
void QFontEngineQPF1::addOutlineToPath(qreal x, qreal y, const QGlyphLayout &glyphs, QPainterPath *path, QTextItem::RenderFlags flags)
{
addBitmapFontToPath(x, y, glyphs, path, flags);
}
glyph_metrics_t QFontEngineQPF1::boundingBox(const QGlyphLayout &glyphs)
{
if (glyphs.numGlyphs == 0)
return glyph_metrics_t();
QFixed w = 0;
for (int i = 0; i < glyphs.numGlyphs; ++i)
w += glyphs.effectiveAdvance(i);
return glyph_metrics_t(0, -ascent(), w, ascent()+descent()+1, w, 0);
}
glyph_metrics_t QFontEngineQPF1::boundingBox(glyph_t glyph)
{
const QPFGlyph *g = d->tree->get(glyph);
if (!g)
return glyph_metrics_t();
Q_ASSERT(g);
return glyph_metrics_t(g->metrics->bearingx, -g->metrics->bearingy,
g->metrics->width, g->metrics->height,
g->metrics->advance, 0);
}
QFixed QFontEngineQPF1::ascent() const
{
return d->fm.ascent;
}
QFixed QFontEngineQPF1::descent() const
{
return d->fm.descent;
}
QFixed QFontEngineQPF1::leading() const
{
return d->fm.leading;
}
qreal QFontEngineQPF1::maxCharWidth() const
{
return d->fm.maxwidth;
}
/*
const char *QFontEngineQPF1::name() const
{
return "qt";
}
*/
bool QFontEngineQPF1::canRender(const QChar *str, int len)
{
for(int i = 0; i < len; i++)
if (!d->tree->inFont(str[i].unicode()))
return false;
return true;
}
QFontEngine::Type QFontEngineQPF1::type() const
{
return QPF1;
}
qreal QFontEngineQPF1::minLeftBearing() const
{
return d->fm.leftbearing;
}
qreal QFontEngineQPF1::minRightBearing() const
{
return d->fm.rightbearing;
}
QFixed QFontEngineQPF1::underlinePosition() const
{
return d->fm.underlinepos;
}
QFixed QFontEngineQPF1::lineThickness() const
{
return d->fm.underlinewidth;
}
#endif //QT_NO_QWS_QPF
QT_END_NAMESPACE