(svn r25570) -Add: cache for ParagraphLayouts.

This commit is contained in:
frosch 2013-07-06 19:00:33 +00:00
parent 520a8de3e1
commit 6bb4748558
3 changed files with 145 additions and 42 deletions

View File

@ -21,6 +21,9 @@
#endif /* WITH_ICU */ #endif /* WITH_ICU */
/** Cache of ParagraphLayout lines. */
Layouter::LineCache Layouter::linecache;
/** Cache of Font instances. */ /** Cache of Font instances. */
Layouter::FontColourMap Layouter::fonts[FS_END]; Layouter::FontColourMap Layouter::fonts[FS_END];
@ -134,6 +137,8 @@ ParagraphLayout *Layouter::GetParagraphLayout(UChar *buff, UChar *buff_end, Font
} }
LEErrorCode status = LE_NO_ERROR; LEErrorCode status = LE_NO_ERROR;
/* ParagraphLayout does not copy "buff", so it must stay valid.
* "runs" is copied according to the ICU source, but the documentation does not specify anything, so this might break somewhen. */
return new ParagraphLayout(buff, length, &runs, NULL, NULL, NULL, _current_text_dir == TD_RTL ? UBIDI_DEFAULT_RTL : UBIDI_DEFAULT_LTR, false, status); return new ParagraphLayout(buff, length, &runs, NULL, NULL, NULL, _current_text_dir == TD_RTL ? UBIDI_DEFAULT_RTL : UBIDI_DEFAULT_LTR, false, status);
} }
@ -277,6 +282,14 @@ ParagraphLayout::ParagraphLayout(WChar *buffer, int length, FontMap &runs) : buf
assert(runs.End()[-1].first == length); assert(runs.End()[-1].first == length);
} }
/**
* Reset the position to the start of the paragraph.
*/
void ParagraphLayout::reflow()
{
this->buffer = this->buffer_begin;
}
/** /**
* Construct a new line with a maximum width. * Construct a new line with a maximum width.
* @param max_width The maximum width of the string. * @param max_width The maximum width of the string.
@ -300,7 +313,7 @@ ParagraphLayout::Line *ParagraphLayout::nextLine(int max_width)
} }
const WChar *begin = this->buffer; const WChar *begin = this->buffer;
WChar *last_space = NULL; const WChar *last_space = NULL;
const WChar *last_char = begin; const WChar *last_char = begin;
int width = 0; int width = 0;
@ -412,62 +425,77 @@ ParagraphLayout *Layouter::GetParagraphLayout(WChar *buff, WChar *buff_end, Font
*/ */
Layouter::Layouter(const char *str, int maxw, TextColour colour, FontSize fontsize) Layouter::Layouter(const char *str, int maxw, TextColour colour, FontSize fontsize)
{ {
const CharType *buffer_last = lastof(this->buffer);
CharType *buff = this->buffer;
FontState state(colour, fontsize); FontState state(colour, fontsize);
WChar c = 0; WChar c = 0;
do { do {
Font *f = GetFont(state.fontsize, state.cur_colour); /* Scan string for end of line */
CharType *buff_begin = buff; const char *lineend = str;
FontMap fontMapping; for (;;) {
size_t len = Utf8Decode(&c, lineend);
if (c == '\0' || c == '\n') break;
lineend += len;
}
/* LineCacheItem& line = GetCachedParagraphLayout(str, lineend - str, state);
* Go through the whole string while adding Font instances to the font map if (line.layout != NULL) {
* whenever the font changes, and convert the wide characters into a format /* Line is in cache */
* usable by ParagraphLayout. str = lineend + 1;
*/ state = line.state_after;
for (; buff < buffer_last;) { line.layout->reflow();
c = Utf8Consume(const_cast<const char **>(&str)); } else {
if (c == '\0' || c == '\n') { /* Line is new, layout it */
break; const CharType *buffer_last = lastof(line.buffer);
} else if (c >= SCC_BLUE && c <= SCC_BLACK) { CharType *buff_begin = line.buffer;
state.SetColour((TextColour)(c - SCC_BLUE)); CharType *buff = buff_begin;
} else if (c == SCC_PREVIOUS_COLOUR) { // Revert to the previous colour. FontMap &fontMapping = line.runs;
state.SetPreviousColour(); Font *f = GetFont(state.fontsize, state.cur_colour);
} else if (c == SCC_TINYFONT) {
state.SetFontSize(FS_SMALL); /*
} else if (c == SCC_BIGFONT) { * Go through the whole string while adding Font instances to the font map
state.SetFontSize(FS_LARGE); * whenever the font changes, and convert the wide characters into a format
} else { * usable by ParagraphLayout.
buff += AppendToBuffer(buff, buffer_last, c); */
continue; for (; buff < buffer_last;) {
c = Utf8Consume(const_cast<const char **>(&str));
if (c == '\0' || c == '\n') {
break;
} else if (c >= SCC_BLUE && c <= SCC_BLACK) {
state.SetColour((TextColour)(c - SCC_BLUE));
} else if (c == SCC_PREVIOUS_COLOUR) { // Revert to the previous colour.
state.SetPreviousColour();
} else if (c == SCC_TINYFONT) {
state.SetFontSize(FS_SMALL);
} else if (c == SCC_BIGFONT) {
state.SetFontSize(FS_LARGE);
} else {
buff += AppendToBuffer(buff, buffer_last, c);
continue;
}
if (!fontMapping.Contains(buff - buff_begin)) {
fontMapping.Insert(buff - buff_begin, f);
}
f = GetFont(state.fontsize, state.cur_colour);
} }
/* Better safe than sorry. */
*buff = '\0';
if (!fontMapping.Contains(buff - buff_begin)) { if (!fontMapping.Contains(buff - buff_begin)) {
fontMapping.Insert(buff - buff_begin, f); fontMapping.Insert(buff - buff_begin, f);
} }
f = GetFont(state.fontsize, state.cur_colour); line.layout = GetParagraphLayout(buff_begin, buff, fontMapping);
line.state_after = state;
} }
/* Better safe than sorry. */
*buff = '\0';
if (!fontMapping.Contains(buff - buff_begin)) {
fontMapping.Insert(buff - buff_begin, f);
}
ParagraphLayout *p = GetParagraphLayout(buff_begin, buff, fontMapping);
/* Copy all lines into a local cache so we can reuse them later on more easily. */ /* Copy all lines into a local cache so we can reuse them later on more easily. */
ParagraphLayout::Line *l; ParagraphLayout::Line *l;
while ((l = p->nextLine(maxw)) != NULL) { while ((l = line.layout->nextLine(maxw)) != NULL) {
*this->Append() = l; *this->Append() = l;
} }
delete p; } while (c != '\0');
} while (c != '\0' && buff < buffer_last);
} }
/** /**
@ -507,4 +535,40 @@ void Layouter::ResetFontCache(FontSize size)
delete it->second; delete it->second;
} }
fonts[size].Clear(); fonts[size].Clear();
/* We must reset the linecache since it references the just freed fonts */
ResetLineCache();
}
/**
* Get reference to cache item.
* If the item does not exist yet, it is default constructed.
* @param str Source string of the line (including colour and font size codes).
* @param len Length of \a str in bytes (no termination).
* @param state State of the font at the beginning of the line.
* @return Reference to cache item.
*/
Layouter::LineCacheItem &Layouter::GetCachedParagraphLayout(const char *str, size_t len, const FontState &state)
{
LineCacheKey key;
key.state_before = state;
key.str.assign(str, len);
return linecache[key];
}
/**
* Clear line cache.
*/
void Layouter::ResetLineCache()
{
linecache.clear();
}
/**
* Reduce the size of linecache if necessary to prevent infinite growth.
*/
void Layouter::ReduceLineCache()
{
/* TODO LRU cache would be fancy, but not exactly necessary */
if (linecache.size() > 4096) ResetLineCache();
} }

View File

@ -16,6 +16,9 @@
#include "gfx_func.h" #include "gfx_func.h"
#include "core/smallmap_type.hpp" #include "core/smallmap_type.hpp"
#include <map>
#include <string>
#ifdef WITH_ICU #ifdef WITH_ICU
#include "layout/ParagraphLayout.h" #include "layout/ParagraphLayout.h"
#define ICU_FONTINSTANCE : public LEFontInstance #define ICU_FONTINSTANCE : public LEFontInstance
@ -32,6 +35,7 @@ struct FontState {
TextColour cur_colour; ///< Current text colour. TextColour cur_colour; ///< Current text colour.
TextColour prev_colour; ///< Text colour from before the last colour switch. TextColour prev_colour; ///< Text colour from before the last colour switch.
FontState() : fontsize(FS_END), cur_colour(TC_INVALID), prev_colour(TC_INVALID) {}
FontState(TextColour colour, FontSize fontsize) : fontsize(fontsize), cur_colour(colour), prev_colour(colour) {} FontState(TextColour colour, FontSize fontsize) : fontsize(fontsize), cur_colour(colour), prev_colour(colour) {}
/** /**
@ -143,10 +147,11 @@ public:
}; };
const WChar *buffer_begin; ///< Begin of the buffer. const WChar *buffer_begin; ///< Begin of the buffer.
WChar *buffer; ///< The current location in the buffer. const WChar *buffer; ///< The current location in the buffer.
FontMap &runs; ///< The fonts we have to use for this paragraph. FontMap &runs; ///< The fonts we have to use for this paragraph.
ParagraphLayout(WChar *buffer, int length, FontMap &runs); ParagraphLayout(WChar *buffer, int length, FontMap &runs);
void reflow();
Line *nextLine(int max_width); Line *nextLine(int max_width);
}; };
#endif /* !WITH_ICU */ #endif /* !WITH_ICU */
@ -166,7 +171,36 @@ class Layouter : public AutoDeleteSmallVector<ParagraphLayout::Line *, 4> {
size_t AppendToBuffer(CharType *buff, const CharType *buffer_last, WChar c); size_t AppendToBuffer(CharType *buff, const CharType *buffer_last, WChar c);
ParagraphLayout *GetParagraphLayout(CharType *buff, CharType *buff_end, FontMap &fontMapping); ParagraphLayout *GetParagraphLayout(CharType *buff, CharType *buff_end, FontMap &fontMapping);
CharType buffer[DRAW_STRING_BUFFER]; ///< Buffer for the text that is going to be drawn. /** Key into the linecache */
struct LineCacheKey {
FontState state_before; ///< Font state at the beginning of the line.
std::string str; ///< Source string of the line (including colour and font size codes).
/** Comparison operator for std::map */
bool operator<(const LineCacheKey &other) const
{
if (this->state_before.fontsize != other.state_before.fontsize) return this->state_before.fontsize < other.state_before.fontsize;
if (this->state_before.cur_colour != other.state_before.cur_colour) return this->state_before.cur_colour < other.state_before.cur_colour;
if (this->state_before.prev_colour != other.state_before.prev_colour) return this->state_before.prev_colour < other.state_before.prev_colour;
return this->str < other.str;
}
};
/** Item in the linecache */
struct LineCacheItem {
/* Stuff that cannot be freed until the ParagraphLayout is freed */
CharType buffer[DRAW_STRING_BUFFER]; ///< Accessed by both ICU's and our ParagraphLayout::nextLine.
FontMap runs; ///< Accessed by our ParagraphLayout::nextLine.
FontState state_after; ///< Font state after the line.
ParagraphLayout *layout; ///< Layout of the line.
LineCacheItem() : layout(NULL) {}
~LineCacheItem() { delete layout; }
};
typedef std::map<LineCacheKey, LineCacheItem> LineCache;
static LineCache linecache;
static LineCacheItem &GetCachedParagraphLayout(const char *str, size_t len, const FontState &state);
typedef SmallMap<TextColour, Font *> FontColourMap; typedef SmallMap<TextColour, Font *> FontColourMap;
static FontColourMap fonts[FS_END]; static FontColourMap fonts[FS_END];
@ -177,6 +211,8 @@ public:
Dimension GetBounds(); Dimension GetBounds();
static void ResetFontCache(FontSize size); static void ResetFontCache(FontSize size);
static void ResetLineCache();
static void ReduceLineCache();
}; };
#endif /* GFX_LAYOUT_H */ #endif /* GFX_LAYOUT_H */

View File

@ -61,6 +61,7 @@
#include "game/game_config.hpp" #include "game/game_config.hpp"
#include "town.h" #include "town.h"
#include "subsidy_func.h" #include "subsidy_func.h"
#include "gfx_layout.h"
#include "linkgraph/linkgraphschedule.h" #include "linkgraph/linkgraphschedule.h"
@ -1318,6 +1319,8 @@ void StateGameLoop()
ClearStorageChanges(false); ClearStorageChanges(false);
Layouter::ReduceLineCache();
if (_game_mode == GM_EDITOR) { if (_game_mode == GM_EDITOR) {
RunTileLoop(); RunTileLoop();
CallVehicleTicks(); CallVehicleTicks();