summaryrefslogtreecommitdiff
path: root/doc/devel/gfxlib
blob: 82ce6b72e70fadcddd64b3788aa489db5249cfe0 (plain)
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
		      The TFB Graphics Libraries

	 Initial documentation by Michael Martin, 4 Feb 2003

The graphics system in UQM has three major subsystems:

- The "legacy" system, which most of the core code uses.  This
  involves the foo_blt () routines in 3do_blt.c, and the data
  types CONTEXT, FRAME, DRAWABLE, and possibly others.  I'm
  less familiar with this code, and want to eradicate as much of it as
  I can, but for now, its lowest level has been rewritten as direct
  calls to:

- The TFB_Draw* commands, documented below in great detail.  These
  routines deal with the datatype TFB_Image, and the more primitive
  TFB_Canvas.  They also support drawing to one of several 'screens',
  but since only one thread should be allowed to touch the screen,
  the TFB_DrawScreen routines end up constructing inputs to:

- The DCQ/DrawCommand library.  This is a ring queue with commands for
  rendering graphics on the actual screen.  It interacts to some
  degree with the CONTEXT datatype, but otherwise is defined entirely
  in terms of TFB_Images and TFB_Canvases.

			  THE LEGACY LIBRARY
			 --------------------

The datatypes the code uses directly are CONTEXT, FRAME, and FONT, all
of which are really pointers to void.  Pointers to those are PCONTEXT,
PFRAME, and PFONT.  Then there's DRAWABLE, which is a DWORD, and its
pointer type PDRAWABLE. (These are defined in sc2code/libs/gfxlib.h.)
I'm not sure how DRAWABLE values are transformed into actual drawable
entities.

The full structures for these are in various files in
sc2code/libs/graphics.

context.h:  defines CONTEXT_DESC and PCONTEXT_DESC (and the equivalent
            CONTEXTPTR -- insert various sorts of incomprenshible
            muttering here),

display.h:  defines a DISPLAY_INTERFACE and PDISPLAY_INTERFACE type (as
            well as a global _pCurDisplay).

drawable.h: defines FRAME_DESC and DRAWABLE_DESC, and the pointer
            types PFRAME_DESC and PDRAWABLE_DESC.  FRAME_DESC has a
            TFB_Image pointer as a member.  DRAWABLE_DESC currently
            still uses a rather annoying technique where the last
            member of a struct is a 1-element array, more memory than
            that is actually allocated, and the array's bounds are
            deliberately overflowed to get at multiple frames.

font.h:     defines FONT_DESC and PFONT_DESC.

(Details on how all these data types work is forthcoming.)

			 THE TFB_DRAW LIBRARY
			----------------------

The TFB_Draw commands have a single header file:
sc2code/libs/graphics/tfb_draw.h.  This file declares the following
data types:

SCREEN: This is an enum, naming the various screens that the DrawCmd
        library can draw to.  Valid values at present are
        TFB_SCREEN_MAIN, TFB_SCREEN_EXTRA, and TFB_SCREEN_TRANSITION.
        These correspond to various objects that are of type TFB_Canvas.

	(The maximal number of screens is provided by a bogus last
        element, TFB_GFX_NUMSCREENS.  Keep that as the last element
        and the allocators will operate properly regardless of any
        screens you may later want to add.)

TFB_Canvas: This is, for the purposes of most of the code, a void
            pointer.  The implementations of TFB_DrawCanvas commands
            cast them to the appropriate type.  (The only
            implementation of these commands casts them to
            SDL_Surface*.)

TFB_Palette: Four UBYTES, r, g, b, and 'unused'.  This is designed at
             present to be directly castable to SDL_Color, which most
             things do.  We should probably do something about that at
             some point.

TFB_Image: The most important image structure.  This has two
           TFB_Canvases (one for the 'core' image, one for a scaled
           version of it), a pointer to a TFB_Palette array, a
           colormap index, a scaling constant, a Mutex from the
           threading library (to ensure internal consistency if
           multiple threads are doing stuff), and a 'dirty bit' which
           means that, if the image is scaled, the ScaledImg needs to
           be recomputed.

TFB_DrawScreen
--------------

When you wish to draw graphics directly on the screen, you call these
routines, which enqueue the DrawCommands:

----

void TFB_DrawScreen_Line (int x1, int y1, int x2, int y2, 
			  int r, int g, int b, 
			  SCREEN dest);

Draws a line from (x1, y1)-(x2, y2) of a color specified by r, g, and
b on the specified screen.

We have a known bug here in that if a corner of one of these lines is
outside of the context's clipping rectangle, the slope of the line may
change.

----

void TFB_DrawScreen_Rect (PRECT rect, 
			  int r, int g, int b, 
			  SCREEN dest);

Draws a (filled) rectangle with the specified color on the destination
screen.  PRECT is part of the legacy library.

void TFB_DrawScreen_Copy (PRECT r, SCREEN src, SCREEN dest);

Copies data between screens.  The PRECT defines the region to copy.
This can be handy in saving and restoring background information.

----

void TFB_DrawScreen_Image (TFB_Image *img, 
			   int x, int y, 
			   BOOLEAN scaled, 
			   TFB_Palette *palette, 
			   SCREEN dest);

void TFB_DrawScreen_FilledImage (TFB_Image *img, 
				 int x, int y, 
				 BOOLEAN scaled, 
				 int r, int g, int b, 
				 SCREEN dest);

These two routines draw images.  img, x, and y are all straightforward
(x and y refer to the upper left of the image), and scaled indicates
whether or not the NormalImg or ScaledImg should be used, and dest
names the target screen.

For TFB_DrawScreen_Image, the 'palette' argument refers to a
256-element array of TFB_Palette that describes the palette to use.
(The default is cached in TFB_Image itself.)  Various techniques are
used to cache the palette values to keep spurious palette-switch
commands from flooding the DrawCmd queue.  However, switching the
palette many times per frame is likely to seriously degrade
performance.

TFB_DrawScreen_FilledImage draws every non-transparent pixel in img in
the color specified by r, g, b.  (Fonts and some menus do this.)

----

void TFB_DrawScreen_WaitForSignal (void);

Puts this thread to sleep until all commands queued to this point are
executed.  Mostly used to keep from spamming the DrawCmd queue, and to
ensure proper operation of the next two routines.

----

void TFB_DrawScreen_CopyToImage (TFB_Image *img, 
				 PRECT lpRect, 
				 SCREEN src);

Load the pixels from the rectangle lpRect in screen src into img.  If
you're actually working with the pixels directly, you'll want to do a
TFB_DrawScreen_WaitForSignal () to ensure the image has actually been
updated.  (If you're just passing it to other TFB_DrawScreen commands,
that's unnecessary, because ordering within a thread is guaranteed.)

----

void TFB_DrawScreen_DeleteImage (TFB_Image *img);

Deallocates all memory associated with img.  This REALLY doesn't
belong here.  It should be a TFB_DrawImage command, with a requirement
that you WaitForSignal lfirst.

----

TFB_DrawImage
-------------

These routines are similar to a subset of the TFB_DrawScreen commands,
except that instead of drawing on the screen at some later time, they
draw directly and immediately onto a TFB_Image.  The arguments all
mean the same things as they did for TFB_DrawScreen.

----

void TFB_DrawImage_Line (int x1, int y1, int x2, int y2, 
                         int r, int g, int b, 
			 TFB_Image *dest);

void TFB_DrawImage_Rect (PRECT rect, 
			 int r, int g, int b, 
			 TFB_Image *image);

void TFB_DrawImage_Image (TFB_Image *img, 
			  int x, int y, 
			  BOOLEAN scaled, 
			  TFB_Palette *palette, 
			  TFB_Image *target);

void TFB_DrawImage_FilledImage (TFB_Image *img, 
				int x, int y, 
				BOOLEAN scaled, 
				int r, int g, int b, 
				TFB_Image *target);

----

TFB_DrawCanvas
--------------

These routines are, quite literally, identical in every way to the
TFB_DrawImage routines, except that they draw on TFB_Canvases instead.
They are defined in graphics-library-specific locations.  There is, at
present, only one implementation of these, in
libs/graphics/sdl/canvas.c.

void TFB_DrawCanvas_Line (int x1, int y1, int x2, int y2, 
			  int r, int g, int b, 
			  TFB_Canvas dest);

void TFB_DrawCanvas_Rect (PRECT rect, 
			  int r, int g, int b, 
			  TFB_Canvas image);

void TFB_DrawCanvas_Image (TFB_Image *img, 
			   int x, int y, 
			   BOOLEAN scaled, 
			   TFB_Palette *palette, 
			   TFB_Canvas target);

void TFB_DrawCanvas_FilledImage (TFB_Image *img, 
				 int x, int y, 
				 BOOLEAN scaled, 
				 int r, int g, int b, 
				 TFB_Canvas target);

----

Creation and Destruction of TFB_Images, TFB_Canvases, and TFB_Palettes
----------------------------------------------------------------------

Various commands exist for creating and destroying the TFB_Draw data
types.  The concept of "ownership" is critical here.  If a data object
owns a pointer inside of it, that pointer's referent is deallocated
when the data object is deallocated.  If a pointer variable owns its
referent, it's permissible to delete it.  

TFB_Canvas and TFB_Palette are primitives.  TFB_Image owns NormalImg,
ScaledImg, and Palette, and will delete them when it is itself
deleted.

That said, here are the routines:

---

TFB_Image *TFB_DrawImage_New (TFB_Canvas canvas)

Creates a new TFB_Image, which the caller then owns.  The caller must
own the canvas, and transfers ownership of that canvas to the image.
The Palette value is automatically created (and the image owns it);
ScaledImg will be NULL until you scale the image and draw it to the
screen.

---

void TFB_DrawImage_Delete (TFB_Image *image)

Deletes the image, and all non-NULL components.  You must own the
image you delete.

---

TFB_Canvas TFB_DrawCanvas_New_TrueColor (int w, int h, BOOLEAN has_alpha);

TFB_Canvas TFB_DrawCanvas_New_Paletted (int w, int h, 
					TFB_Palette *palette, 
					int transparent_index);

These create new TFB_Canvases, which the caller will then own.  Width
and height are straightforward.  The TrueColor variant produces Canvases
with the same color depth and pixel format as the screen.  The
has_alpha flag indicates whether or not the canvas has an alpha
channel.

The Paletted variant produces 8-bit paletted canvases.  The palette
argument is optional (it can be NULL, in which case you'll need to set
it later - TFB_Images tend to do this when drawn), as is the
transparent_index (if -1, there is no transparency; otherwise, it's
the index of the transparent color).

---

TFB_Canvas TFB_DrawCanvas_ToScreenFormat (TFB_Canvas canvas);

Returns a canvas that, if possible, matches the graphics configuration
of the screen.  You must own the source canvas.  If the conversion is
possible, it makes the conversion, deletes its argument, and returns
the converted version; if conversion is not possible, the canvas is
returned intact.

Regardless of success or failure, the caller owns the result.

---

TFB_Palette *TFB_DrawCanvas_ExtractPalette (TFB_Canvas canvas);

Allocates and returns a 256-entry TFB_Palette array that describes the
palette of the canvas, or returns NULL if the canvas is true-color.
If the result is non-NULL, the caller owns the result.  The caller
need not own canvas.


			   DRAWCMD LIBRARY
			  -----------------

Documentation yet to be written.  UQM code shouldn't really mess with
the DrawCmd library directly.