1 |
root |
1.1 |
typedef struct { |
2 |
|
|
GLenum mode; |
3 |
root |
1.3 |
GLenum format; // 0, GL_T2F_V3F, GL_V2F |
4 |
root |
1.7 |
GLuint texname; |
5 |
root |
1.1 |
unsigned char r, g, b, a; |
6 |
|
|
} rc_key_t; |
7 |
|
|
|
8 |
|
|
typedef struct { |
9 |
|
|
HV *hv; |
10 |
|
|
} rc_t; |
11 |
|
|
|
12 |
|
|
typedef SV rc_array_t; |
13 |
|
|
|
14 |
|
|
static rc_t * |
15 |
root |
1.8 |
rc_alloc (void) |
16 |
root |
1.1 |
{ |
17 |
|
|
rc_t *rc = g_slice_new0 (rc_t); |
18 |
|
|
rc->hv = newHV (); |
19 |
|
|
|
20 |
|
|
return rc; |
21 |
|
|
} |
22 |
|
|
|
23 |
|
|
static void |
24 |
|
|
rc_free (rc_t *rc) |
25 |
|
|
{ |
26 |
|
|
SvREFCNT_dec (rc->hv); |
27 |
|
|
g_slice_free (rc_t, rc); |
28 |
|
|
} |
29 |
|
|
|
30 |
|
|
static void |
31 |
|
|
rc_clear (rc_t *rc) |
32 |
|
|
{ |
33 |
|
|
hv_clear (rc->hv); |
34 |
|
|
} |
35 |
|
|
|
36 |
|
|
static rc_array_t * |
37 |
|
|
rc_array (rc_t *rc, rc_key_t *k) |
38 |
|
|
{ |
39 |
|
|
SV *sv = *hv_fetch (rc->hv, (char *)k, sizeof (*k), 1); |
40 |
|
|
|
41 |
|
|
if (SvTYPE (sv) != SVt_PV) |
42 |
|
|
{ |
43 |
|
|
sv_upgrade (sv, SVt_PV); |
44 |
|
|
SvPOK_only (sv); |
45 |
|
|
} |
46 |
|
|
|
47 |
|
|
return sv; |
48 |
|
|
} |
49 |
|
|
|
50 |
|
|
static void |
51 |
|
|
rc_v2f (rc_array_t *arr, float x, float y) |
52 |
|
|
{ |
53 |
root |
1.2 |
STRLEN len = SvCUR (arr) + sizeof (float) * 2; |
54 |
root |
1.1 |
SvGROW (arr, len); |
55 |
|
|
|
56 |
|
|
((float *)SvEND (arr))[0] = x; |
57 |
|
|
((float *)SvEND (arr))[1] = y; |
58 |
|
|
|
59 |
|
|
SvCUR_set (arr, len); |
60 |
|
|
} |
61 |
|
|
|
62 |
|
|
static void |
63 |
|
|
rc_t2f_v3f (rc_array_t *arr, float u, float v, float x, float y, float z) |
64 |
|
|
{ |
65 |
|
|
STRLEN len = SvCUR (arr) + sizeof (float) * 5; |
66 |
|
|
SvGROW (arr, len); |
67 |
|
|
|
68 |
|
|
((float *)SvEND (arr))[0] = u; |
69 |
|
|
((float *)SvEND (arr))[1] = v; |
70 |
|
|
|
71 |
|
|
((float *)SvEND (arr))[2] = x; |
72 |
|
|
((float *)SvEND (arr))[3] = y; |
73 |
|
|
((float *)SvEND (arr))[4] = z; |
74 |
|
|
|
75 |
|
|
SvCUR_set (arr, len); |
76 |
|
|
} |
77 |
|
|
|
78 |
|
|
static void |
79 |
root |
1.4 |
rc_glyph (rc_array_t *arr, int u, int v, int w, int h, int x, int y) |
80 |
|
|
{ |
81 |
root |
1.9 |
if (w && h) |
82 |
|
|
{ |
83 |
|
|
U8 *c; |
84 |
|
|
STRLEN len = SvCUR (arr); |
85 |
|
|
SvGROW (arr, len + 2 * 2 + 1 * 4); |
86 |
|
|
c = (U8 *)SvEND (arr); |
87 |
|
|
|
88 |
|
|
x += w; |
89 |
|
|
y += h; |
90 |
|
|
|
91 |
|
|
*c++ = u; |
92 |
|
|
*c++ = v; |
93 |
|
|
*c++ = w; |
94 |
|
|
*c++ = h; |
95 |
|
|
|
96 |
|
|
// use ber-encoding for up to 14 bits (16k) |
97 |
|
|
*c = 0x80 | (x >> 7); c += (x >> 7) ? 1 : 0; *c++ = x & 0x7f; |
98 |
|
|
*c = 0x80 | (y >> 7); c += (y >> 7) ? 1 : 0; *c++ = y & 0x7f; |
99 |
root |
1.4 |
|
100 |
root |
1.9 |
SvCUR_set (arr, c - (U8 *)SvPVX (arr)); |
101 |
|
|
} |
102 |
root |
1.4 |
} |
103 |
|
|
|
104 |
|
|
static void |
105 |
root |
1.1 |
rc_draw (rc_t *rc) |
106 |
|
|
{ |
107 |
|
|
HE *he; |
108 |
|
|
|
109 |
|
|
hv_iterinit (rc->hv); |
110 |
|
|
while ((he = hv_iternext (rc->hv))) |
111 |
|
|
{ |
112 |
|
|
rc_key_t *key = (rc_key_t *)HeKEY (he); |
113 |
|
|
rc_array_t *arr = HeVAL (he); |
114 |
|
|
STRLEN len; |
115 |
|
|
char *arr_pv = SvPV (arr, len); |
116 |
|
|
GLsizei stride; |
117 |
|
|
|
118 |
|
|
if (key->texname) |
119 |
|
|
{ |
120 |
|
|
glBindTexture (GL_TEXTURE_2D, key->texname); |
121 |
|
|
glEnable (GL_TEXTURE_2D); |
122 |
|
|
} |
123 |
|
|
else |
124 |
|
|
glDisable (GL_TEXTURE_2D); |
125 |
|
|
|
126 |
|
|
glColor4ub (key->r, key->g, key->b, key->a); |
127 |
root |
1.3 |
|
128 |
root |
1.4 |
if (key->format) |
129 |
|
|
{ |
130 |
|
|
stride = key->format == GL_T2F_V3F ? sizeof (float) * 5 |
131 |
|
|
: key->format == GL_V2F ? sizeof (float) * 2 |
132 |
|
|
: 65536; |
133 |
|
|
|
134 |
|
|
glInterleavedArrays (key->format, 0, (void *)arr_pv); |
135 |
|
|
//glLockArraysEXT (0, len / stride); |
136 |
|
|
glDrawArrays (key->mode, 0, len / stride); |
137 |
|
|
//glUnlockArraysEXT (); |
138 |
|
|
} |
139 |
|
|
else |
140 |
|
|
{ |
141 |
|
|
// optimised character quad storage. slower but nice on memory. |
142 |
|
|
// reduces storage requirements from 80 bytes/char to 6-8 |
143 |
|
|
U8 *c = (U8 *)arr_pv; |
144 |
|
|
U8 *e = c + len; |
145 |
|
|
|
146 |
|
|
glBegin (key->mode); // practically must be quads |
147 |
|
|
|
148 |
|
|
while (c < e) |
149 |
|
|
{ |
150 |
|
|
int u, v, x, y, w, h; |
151 |
|
|
|
152 |
|
|
u = *c++; |
153 |
|
|
v = *c++; |
154 |
|
|
w = *c++; |
155 |
|
|
h = *c++; |
156 |
|
|
|
157 |
|
|
x = *c++; if (x > 0x7f) x = ((x & 0x7f) << 7) | *c++; |
158 |
|
|
y = *c++; if (y > 0x7f) y = ((y & 0x7f) << 7) | *c++; |
159 |
|
|
|
160 |
root |
1.6 |
x -= w; |
161 |
|
|
y -= h; |
162 |
|
|
|
163 |
root |
1.4 |
glTexCoord2f ( u * (1.f / TC_WIDTH), v * (1.f / TC_HEIGHT)); glVertex2i (x , y ); |
164 |
|
|
glTexCoord2f ((u + w) * (1.f / TC_WIDTH), v * (1.f / TC_HEIGHT)); glVertex2i (x + w, y ); |
165 |
|
|
glTexCoord2f ((u + w) * (1.f / TC_WIDTH), (v + h) * (1.f / TC_HEIGHT)); glVertex2i (x + w, y + h); |
166 |
|
|
glTexCoord2f ( u * (1.f / TC_WIDTH), (v + h) * (1.f / TC_HEIGHT)); glVertex2i (x , y + h); |
167 |
|
|
} |
168 |
root |
1.3 |
|
169 |
root |
1.4 |
glEnd (); |
170 |
|
|
} |
171 |
root |
1.1 |
} |
172 |
|
|
|
173 |
|
|
glDisable (GL_TEXTURE_2D); |
174 |
|
|
} |
175 |
|
|
|
176 |
|
|
|
177 |
|
|
|
178 |
|
|
|