all repos — openbox @ 4947902d269213edee40f3f31f97721fa0dd3877

openbox fork - make it a bit more like ryudo

otk/gccache.hh (raw)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
// -*- mode: C++; indent-tabs-mode: nil; c-basic-offset: 2; -*-
#ifndef __gccache_hh
#define __gccache_hh

extern "C" {
#include <X11/Xlib.h>
}

#include "display.hh"
#include "color.hh"

namespace otk {

class GCCacheItem;

class GCCacheContext {
public:
  void set(const Color &_color, const XFontStruct * const _font,
           const int _function, const int _subwindow, const int _linewidth);
  void set(const XFontStruct * const _font);

  ~GCCacheContext(void);

private:
  GCCacheContext()
    : gc(0), pixel(0ul), fontid(0ul),
      function(0), subwindow(0), used(false), screen(~(0u)), linewidth(0) {}

  GC gc;
  unsigned long pixel;
  unsigned long fontid;
  int function;
  int subwindow;
  bool used;
  unsigned int screen;
  int linewidth;

  GCCacheContext(const GCCacheContext &_nocopy);
  GCCacheContext &operator=(const GCCacheContext &_nocopy);

  friend class GCCache;
  friend class GCCacheItem;
};

class GCCacheItem {
public:
  inline const GC &gc(void) const { return ctx->gc; }

private:
  GCCacheItem(void) : ctx(0), count(0), hits(0), fault(false) { }

  GCCacheContext *ctx;
  unsigned int count;
  unsigned int hits;
  bool fault;

  GCCacheItem(const GCCacheItem &_nocopy);
  GCCacheItem &operator=(const GCCacheItem &_nocopy);

  friend class GCCache;
};

class GCCache {
public:
  GCCache(unsigned int screen_count);
  ~GCCache(void);

  // cleans up the cache
  void purge(void);

  GCCacheItem *find(const Color &_color, const XFontStruct * const _font = 0,
                     int _function = GXcopy, int _subwindow = ClipByChildren,
                     int _linewidth = 0);
  void release(GCCacheItem *_item);

private:
  GCCacheContext *nextContext(unsigned int _screen);
  void release(GCCacheContext *ctx);

  // this is closely modelled after the Qt GC cache, but with some of the
  // complexity stripped out
  const unsigned int context_count;
  const unsigned int cache_size;
  const unsigned int cache_buckets;
  const unsigned int cache_total_size;
  GCCacheContext **contexts;
  GCCacheItem **cache;
};

class Pen {
public:
  inline Pen(const Color &_color,  const XFontStruct * const _font = 0,
              int _linewidth = 0, int _function = GXcopy,
              int _subwindow = ClipByChildren)
    : color(_color), font(_font), linewidth(_linewidth), function(_function),
      subwindow(_subwindow), cache(Display::gcCache()), item(0) { }

  inline ~Pen(void) { if (item) cache->release(item); }

  inline const GC &gc(void) const {
    if (! item) item = cache->find(color, font, function, subwindow,
                                   linewidth);
    return item->gc();
  }

private:
  const Color &color;
  const XFontStruct *font;
  int linewidth;
  int function;
  int subwindow;

  mutable GCCache *cache;
  mutable GCCacheItem *item;
};

}

#endif // __gccache_hh